Entity relationship diagrams

The “crow’s foot” that connects the tables together shows us how the columns in one table relate to the columns in another table. ## Types Of Statements A few statements include: * CREATE TABLE is a statement that creates a new table in a database. * DROP TABLE is a statement that removes a table in a database. * SELECT allows you to read data and display it. This is called a query.

SELECT & FROM Statements

Here you were introduced to two statements that will be used in every query you write * SELECT - what columns you want back. * FROM - what table you are querying from.

SELECT id, account_id, occurred_at
FROM orders;

LIMIT Statement

The LIMIT statement is useful when you want to see just the first few rows of a table. This can be much faster for loading than if we load the entire dataset. The LIMIT command is always the very last part of a query. An example of showing just the first 10 rows of the orders table with all of the columns might look like the following:

SELECT *
FROM orders
LIMIT 10

ORDER BY Statement

SELECT id, occurred_at, total_amt_usd
FROM orders
ORDER BY occurred_at
LIMIT 10;

We can ORDER BY more than one column at a time. The statement sorts according to columns listed from left first and those listed on the right after that.

SELECT *
FROM orders
ORDER BY occurred_at, total_amt_usd
LIMIT 10;

WHERE Statements

Using the WHERE statement, we can subset out tables based on conditions that must be met.

SELECT *
FROM orders
WHERE gloss_amt_usd >= 1000
LIMIT 5;

The WHERE statement can also be used with non-numerical data. We can use the = and != operators here. You also need to be sure to use single quotes

SELECT name, website, primary_poc
FROM accounts
WHERE name = 'Exxon Mobil';

Arithmetic Operators

Derived Columns

Creating a new column that is a combination of existing columns is known as a derived column. ### Order of Operations Remember BOMDAS from maths class? The same order of operations apply when using arithmetic operators in SQL.

SELECT id, account_id, 
       poster_amt_usd/(standard_amt_usd + gloss_amt_usd + poster_amt_usd) AS post_per
FROM orders;

Introduction to Logical Operators

LIKE - This allows you to perform operations similar to using WHERE and =, but for cases when you might not know exactly what you are looking for. The LIKE operator is frequently used with %. The % tells us that we might want any number of characters leading up to a particular set of characters or following a certain set of characters.

/* select all the companies whose names start with 'C' */
SELECT name 
FROM accounts
WHERE name like 'C%'

/* select All companies whose names contain the string 'one' somewhere in the name. */
SELECT name
FROM accounts
WHERE name LIKE '%one%';

IN - This allows you to perform operations similar to using WHERE and =, but for more than one condition. We can check one, two or many column values for which we want to pull data, but all within the same query. TIP - In most SQL environments, you can use single or double quotation marks - and you may NEED to use double quotation marks if you have an apostrophe within the text you are attempting to pull.

SELECT name, primary_poc, sales_rep_id
FROM accounts
WHERE name IN ('Walmart', 'Target', 'Nordstrom');

NOT - This is used with IN and LIKE to select all of the rows NOT LIKE or NOT IN a certain condition.

/* Use the accounts table to find the account name, primary poc, and sales rep id for all stores except Walmart, Target, and Nordstrom */
SELECT name, primary_poc, sales_rep_id
FROM accounts
WHERE name NOT IN ('Walmart', 'Target', 'Nordstrom');

AND & BETWEEN - These allow you to combine operations where all combined conditions must be true. * The AND operator is used within a WHERE statement to consider more than one logical clause at a time. Each time you link a new statement with an AND, you will need to specify the column you are interested in looking at. * Sometimes we can make a cleaner statement using BETWEEN than we can using AND. Particularly this is true when we are using the same column for different parts of our AND statement. e.g.:

/*Instead of writing : WHERE column >= 6 AND column <= 10 we can instead write*/ 
WHERE column BETWEEN 6 AND 10
/* Write a query that returns all the orders where the standard_qty is over 1000, the poster_qty is 0, and the gloss_qty is 0 */
SELECT *
FROM orders
WHERE standard_qty > 1000 AND poster_qty = 0 AND gloss_qty = 0;

/* Using the accounts table find all the companies whose names do not start with 'C' and end with 's' */
SELECT name
FROM accounts
WHERE name NOT LIKE 'C%' AND name LIKE '%s';

/* Use the web_events table to find all information regarding individuals who were contacted via organic or adwords and started their account at any point in 2016 sorted from newest to oldest */ 
SELECT *
FROM web_events
WHERE channel IN ('organic', 'adwords') AND occurred_at BETWEEN '2016-01-01' AND '2017-01-01'
ORDER BY occurred_at DESC;
/*  You will notice that using BETWEEN is tricky for dates! While BETWEEN is generally inclusive of endpoints, it assumes the time is at 00:00:00 (i.e. midnight) for dates. This is the reason why we set the right-side endpoint of the period at '2017-01-01'. */

OR - This allow you to combine operations where at least one of the combined conditions must be true. Similar to the AND operator, the OR operator can combine multiple statements. Each time you link a new statement with an OR, you will need to specify the column you are interested in looking at.

/* Find list of orders ids where either gloss_qty or poster_qty is greater than 4000. Only include the id field in the resulting table. */
SELECT id
FROM orders
WHERE gloss_qty > 4000 OR poster_qty > 4000;

/* Write a query that returns a list of orders where the standard_qty is zero and either the gloss_qty or poster_qty is over 1000. */
SELECT *
FROM orders
WHERE standard_qty = 0 AND (gloss_qty > 1000 OR poster_qty > 1000);

/* Find all the company names that start with a 'C' or 'W', and the primary contact contains 'ana' or 'Ana', but it doesn't contain 'eana'*/
SELECT *
FROM accounts
WHERE (name LIKE 'C%' OR name LIKE 'W%') 
           AND ((primary_poc LIKE '%ana%' OR primary_poc LIKE '%Ana%') 
           AND primary_poc NOT LIKE '%eana%');

Summary of basic comands:


JOINS

Inner joins

Only return rows that appear in both tables e.g. Using ERD: Pull standard_qty, gloss_qty, and poster_qty from the orders table, and the website and the primary_poc from the accounts table.

SELECT orders.standard_qty, orders.gloss_qty, 
       orders.poster_qty,  accounts.website, 
       accounts.primary_poc
FROM orders
JOIN accounts
ON orders.account_id = accounts.id

Notice that we need to specify every table a column comes from in the SELECT statement. ### ALIAS When we JOIN tables together, it is nice to give each table an alias. Frequently an alias is just the first letter of the table name.

FROM tablename t1 
JOIN tablename2 t2

Questions

  1. Provide a table for all web_events associated with account name of Walmart. There should be three columns. Be sure to include the primary_poc, time of the event, and the channel for each event. Additionally, you might choose to add a fourth column to assure only Walmart events were chosen.
SELECT a.primary_poc, w.occurred_at, w.channel, a.name
FROM web_events w
JOIN accounts a
ON w.account_id = a.id
WHERE a.name = 'Walmart';
  1. Provide a table that provides the region for each sales_rep along with their associated accounts. Your final table should include three columns: the region name, the sales rep name, and the account name. Sort the accounts alphabetically (A-Z) according to account name.
SELECT r.name region, s.name rep, a.name account
FROM sales_reps s
JOIN region r
ON s.region_id = r.id
JOIN accounts a
ON a.sales_rep_id = s.id
ORDER BY a.name;
  1. Provide the name for each region for every order, as well as the account name and the unit price they paid (total_amt_usd/total) for the order. Your final table should have 3 columns: region name, account name, and unit price. A few accounts have 0 for total, so I divided by (total + 0.01) to assure not dividing by zero.
SELECT r.name region, a.name account, 
       o.total_amt_usd/(o.total + 0.01) unit_price
FROM region r
JOIN sales_reps s
ON s.region_id = r.id
JOIN accounts a
ON a.sales_rep_id = s.id
JOIN orders o
ON o.account_id = a.id;

Outer joins

If we want to include data that only exists in one of the tables we ant to join we can use an outer join If there is not matching information in the JOINed table, then you will have columns with empty cells with a data type called NULL. Writing a LEFT join statement: Comparing LEFT and RIGHT join: LEFT and RIGHT joins are essentially inter changeable so its standard to just use LEFT joins.

JOINS and filtering

If you wanted to filter your results so that it only showed sales from a particular sales rep you could use WHERE but it’s better to use AND as part of the ON statement. A simple rule to remember this is that, when the database executes this query, it executes the join and everything in the ON clause first. Think of this as building the new result set. That result set is then filtered using the WHERE clause.

SELECT orders.*, 
accounts.*
FROM orders
LEFT JOIN accounts
ON orders.account_id = accounts.id
/* Use AND in stead of WHERE for filtering this LEFT JOIN */
AND accounts.sales_rep_id = 325 

Questions

  1. Provide a table that provides the region for each sales_rep along with their associated accounts. This time only for the Midwest region. Your final table should include three columns: the region name, the sales rep name, and the account name. Sort the accounts alphabetically (A-Z) according to account name.
SELECT r.name region, s.name rep, a.name account
FROM sales_reps s
JOIN region r
ON s.region_id = r.id
JOIN accounts a
ON a.sales_rep_id = s.id
WHERE r.name = 'Midwest'
ORDER BY a.name;
  1. Provide a table that provides the region for each sales_rep along with their associated accounts. This time only for accounts where the sales rep has a first name starting with S and in the Midwest region. Your final table should include three columns: the region name, the sales rep name, and the account name. Sort the accounts alphabetically (A-Z) according to account name.
SELECT r.name region, s.name rep, a.name account
FROM sales_reps s
JOIN region r
ON s.region_id = r.id
JOIN accounts a
ON a.sales_rep_id = s.id
WHERE r.name = 'Midwest' AND s.name LIKE 'S%'
ORDER BY a.name;
  1. Provide a table that provides the region for each sales_rep along with their associated accounts. This time only for accounts where the sales rep has a last name starting with K and in the Midwest region. Your final table should include three columns: the region name, the sales rep name, and the account name. Sort the accounts alphabetically (A-Z) according to account name.
SELECT r.name region, s.name rep, a.name account
FROM sales_reps s
JOIN region r
ON s.region_id = r.id
JOIN accounts a
ON a.sales_rep_id = s.id
WHERE r.name = 'Midwest' AND s.name LIKE '% K%'
ORDER BY a.name;
  1. Provide the name for each region for every order, as well as the account name and the unit price they paid (total_amt_usd/total) for the order. However, you should only provide the results if the standard order quantity exceeds 100. Your final table should have 3 columns: region name, account name, and unit price. In order to avoid a division by zero error, adding .01 to the denominator here is helpful total_amt_usd/(total+0.01).
SELECT r.name region, a.name account, o.total_amt_usd/(o.total + 0.01) unit_price
FROM region r
JOIN sales_reps s
ON s.region_id = r.id
JOIN accounts a
ON a.sales_rep_id = s.id
JOIN orders o
ON o.account_id = a.id
WHERE o.standard_qty > 100;
  1. Provide the name for each region for every order, as well as the account name and the unit price they paid (total_amt_usd/total) for the order. However, you should only provide the results if the standard order quantity exceeds 100 and the poster order quantity exceeds 50. Your final table should have 3 columns: region name, account name, and unit price. Sort for the smallest unit price first. In order to avoid a division by zero error, adding .01 to the denominator here is helpful (total_amt_usd/(total+0.01).
SELECT r.name region, a.name account, o.total_amt_usd/(o.total + 0.01) unit_price
FROM region r
JOIN sales_reps s
ON s.region_id = r.id
JOIN accounts a
ON a.sales_rep_id = s.id
JOIN orders o
ON o.account_id = a.id
WHERE o.standard_qty > 100 AND o.poster_qty > 50
ORDER BY unit_price;
  1. Provide the name for each region for every order, as well as the account name and the unit price they paid (total_amt_usd/total) for the order. However, you should only provide the results if the standard order quantity exceeds 100 and the poster order quantity exceeds 50. Your final table should have 3 columns: region name, account name, and unit price. Sort for the largest unit price first. In order to avoid a division by zero error, adding .01 to the denominator here is helpful (total_amt_usd/(total+0.01).
SELECT r.name region, a.name account, o.total_amt_usd/(o.total + 0.01) unit_price
FROM region r
JOIN sales_reps s
ON s.region_id = r.id
JOIN accounts a
ON a.sales_rep_id = s.id
JOIN orders o
ON o.account_id = a.id
WHERE o.standard_qty > 100 AND o.poster_qty > 50
ORDER BY unit_price DESC;
  1. What are the different channels used by account id 1001? Your final table should have only 2 columns: account name and the different channels. You can try SELECT DISTINCT to narrow down the results to only the unique values.
SELECT DISTINCT a.name, w.channel
FROM accounts a
RIGHT JOIN web_events w
ON a.id = w.account_id
WHERE a.id = '1001';
  1. Find all the orders that occurred in 2015. Your final table should have 4 columns: occurred_at, account name, order total, and order total_amt_usd.
SELECT o.occurred_at, a.name, o.total, o.total_amt_usd
FROM accounts a
JOIN orders o
ON o.account_id = a.id
WHERE o.occurred_at BETWEEN '01-01-2015' AND '01-01-2016'
ORDER BY o.occurred_at DESC;

AGGREGATIONS

Null values

When identifying NULLs in a WHERE clause, we write IS NULL or IS NOT NULL. We don’t use =, because NULL isn’t considered a value in SQL.

COUNT

Count the number of rows in a table

SELECT COUNT(*)
FROM accounts;

COUNT does not consider rows that have NULL values. Therefore, this can be useful for quickly identifying which rows have missing data.

SUM

Unlike COUNT, you can only use SUM on numeric columns. However, SUM will ignore NULL values and treat them as 0.

Questions

  1. Find the total amount of poster_qty paper ordered in the orders table.
SELECT SUM(poster_qty) AS total_poster_sales
FROM orders;
  1. Find the total amount of standard_qty paper ordered in the orders table.
SELECT SUM(standard_qty) AS total_standard_sales
FROM orders;
  1. Find the total dollar amount of sales using the total_amt_usd in the orders table.
SELECT SUM(total_amt_usd) AS total_dollar_sales
FROM orders;
  1. Find the total amount spent on standard_amt_usd and gloss_amt_usd paper for each order in the orders table. This should give a dollar amount for each order in the table. Notice, this solution did not use an aggregate.
SELECT standard_amt_usd + gloss_amt_usd AS total_standard_gloss
FROM orders;
  1. Find the standard_amt_usd per unit of standard_qty paper. Your solution should use both an aggregation and a mathematical operator. Notice, this solution used both an aggregate and our mathematical operators
SELECT SUM(standard_amt_usd)/SUM(standard_qty) AS standard_price_per_unit
FROM orders;

MIN and MAX

MIN and MAX are aggregators that ignore NULL values. Functionally, MIN and MAX are similar to COUNT in that they can be used on non-numerical columns. Depending on the column type, MIN will return the lowest number, earliest date, or non-numerical value as early in the alphabet as possible. As you might suspect, MAX does the opposite—it returns the highest number, the latest date, or the non-numerical value closest alphabetically to “Z.”

Questions

  1. When was the earliest order ever placed?
SELECT MIN(occurred_at) 
FROM orders;

Try performing the same query as in question 1 without using an aggregation function.

SELECT occurred_at 
FROM orders 
ORDER BY occurred_at
LIMIT 1;
  1. When did the most recent (latest) web_event occur?
SELECT MAX(occurred_at)
FROM web_events;

Try to perform the result of the previous query without using an aggregation function.

SELECT occurred_at
FROM web_events
ORDER BY occurred_at DESC
LIMIT 1;

AVG

AVG returns the mean of the data - that is the sum of all of the values in the column divided by the number of values in a column. This aggregate function ignores the NULL values in both the numerator and the denominator.

Questions

  1. Find the mean (AVERAGE) amount spent per order on each paper type, as well as the mean amount of each paper type purchased per order. Your final answer should have 6 values - one for each paper type for the average number of sales, as well as the average amount.
SELECT AVG(standard_qty) mean_standard, AVG(gloss_qty) mean_gloss, 
           AVG(poster_qty) mean_poster, AVG(standard_amt_usd) mean_standard_usd, 
           AVG(gloss_amt_usd) mean_gloss_usd, AVG(poster_amt_usd) mean_poster_usd
FROM orders;
  1. What is the MEDIAN total_usd spent on all orders? Note, this is more advanced than the topics we have covered thus far to build a general solution, but we can hard code a solution in the following way.
SELECT *
FROM (SELECT total_amt_usd
      FROM orders
      ORDER BY total_amt_usd
      LIMIT 3457) AS Table1
ORDER BY total_amt_usd DESC
LIMIT 2;

Since there are 6912 orders - we want the average of the 3457 and 3456 order amounts when ordered. This is the average of 2483.16 and 2482.55. This gives the median of 2482.855. This obviously isn’t an ideal way to compute. If we obtain new orders, we would have to change the limit. SQL didn’t even calculate the median for us. The above used a SUBQUERY, but you could use any method to find the two necessary values, and then you just need the average of them.

GROUP BY

GROUP BY can be used to aggregate data within subsets of the data. For example, grouping for different accounts, different regions, or different sales representatives. Any column in the SELECT statement that is not within an aggregator must be in the GROUP BY clause. The GROUP BY always goes between WHERE and ORDER BY. ORDER BY works like SORT in spreadsheet software. SQL evaluates the aggregations before the LIMIT clause. If you don’t group by any columns, you’ll get a 1-row result—no problem there. If you group by a column with enough unique values that it exceeds the LIMIT number, the aggregates will be calculated, and then some rows will simply be omitted from the results. This is a nice way to do things because you know you’re going to get the correct aggregates. If SQL cuts the table down to 100 rows, then performed the aggregations, your results would be substantially different.

Note: You can reference the columns in your select statement in GROUP BY and ORDER BY clauses with numbers that follow the order they appear in the select statement. For example

SELECT standard_qty, COUNT(*)

FROM orders

GROUP BY 1 /*(this 1 refers to standard_qty since it is the first of the columns included in the select statement)*/

ORDER BY 1 /*(this 1 refers to standard_qty since it is the first of the columns included in the select statement)*/

Questions

  1. Which account (by name) placed the earliest order? Your solution should have the account name and the date of the order.
SELECT a.name, o.occurred_at
FROM accounts a
JOIN orders o
ON a.id = o.account_id
ORDER BY occurred_at
LIMIT 1;
  1. Find the total sales in usd for each account. You should include two columns - the total sales for each company’s orders in usd and the company name.
SELECT a.name, SUM(total_amt_usd) total_sales
FROM orders o
JOIN accounts a
ON a.id = o.account_id
GROUP BY a.name;
  1. Via what channel did the most recent (latest) web_event occur, which account was associated with this web_event? Your query should return only three values - the date, channel, and account name.
SELECT w.occurred_at, w.channel, a.name
FROM web_events w
JOIN accounts a
ON w.account_id = a.id 
ORDER BY w.occurred_at DESC
LIMIT 1;
  1. Find the total number of times each type of channel from the web_events was used. Your final table should have two columns - the channel and the number of times the channel was used.
SELECT w.channel, COUNT(*)
FROM web_events w
GROUP BY w.channel
  1. Who was the primary contact associated with the earliest web_event?
SELECT a.primary_poc
FROM web_events w
JOIN accounts a
ON a.id = w.account_id
ORDER BY w.occurred_at
LIMIT 1;
  1. What was the smallest order placed by each account in terms of total usd. Provide only two columns - the account name and the total usd. Order from smallest dollar amounts to largest.
SELECT a.name, MIN(total_amt_usd) smallest_order
FROM accounts a
JOIN orders o
ON a.id = o.account_id
GROUP BY a.name
ORDER BY smallest_order;

Sort of strange we have a bunch of orders with no dollars. We might want to look into those.

  1. Find the number of sales reps in each region. Your final table should have two columns - the region and the number of sales_reps. Order from fewest reps to most reps.
SELECT r.name, COUNT(*) num_reps
FROM region r
JOIN sales_reps s
ON r.id = s.region_id
GROUP BY r.name
ORDER BY num_reps;

You can GROUP BY multiple columns at once. This is often useful to aggregate across a number of different segments. The order of columns listed in the ORDER BY clause does make a difference. You are ordering the columns from left to right. The order of column names in your GROUP BY clause doesn’t matter—the results will be the same regardless.

Questions

  1. For each account, determine the average amount of each type of paper they purchased across their orders. Your result should have four columns - one for the account name and one for the average spent on each of the paper types.
SELECT a.name, AVG(o.standard_qty) avg_stand, AVG(o.gloss_qty) avg_gloss, AVG(o.poster_qty) avg_post
FROM accounts a
JOIN orders o
ON a.id = o.account_id
GROUP BY a.name;
  1. For each account, determine the average amount spent per order on each paper type. Your result should have four columns - one for the account name and one for the average amount spent on each paper type.
SELECT a.name, AVG(o.standard_amt_usd) avg_stand, AVG(o.gloss_amt_usd) avg_gloss, AVG(o.poster_amt_usd) avg_post
FROM accounts a
JOIN orders o
ON a.id = o.account_id
GROUP BY a.name;
  1. Determine the number of times a particular channel was used in the web_events table for each sales rep. Your final table should have three columns - the name of the sales rep, the channel, and the number of occurrences. Order your table with the highest number of occurrences first.
SELECT s.name, w.channel, COUNT(*) num_events
FROM accounts a
JOIN web_events w
ON a.id = w.account_id
JOIN sales_reps s
ON s.id = a.sales_rep_id
GROUP BY s.name, w.channel
ORDER BY num_events DESC;
  1. Determine the number of times a particular channel was used in the web_events table for each region. Your final table should have three columns - the region name, the channel, and the number of occurrences. Order your table with the highest number of occurrences first.
SELECT r.name, w.channel, COUNT(*) num_events
FROM accounts a
JOIN web_events w
ON a.id = w.account_id
JOIN sales_reps s
ON s.id = a.sales_rep_id
JOIN region r
ON r.id = s.region_id
GROUP BY r.name, w.channel
ORDER BY num_events DESC;

DISTINCT

DISTINCT is always used in SELECT statements, and it provides the unique rows for all columns written in the SELECT statement. Therefore, you only use DISTINCT once in any particular SELECT statement.

SELECT DISTINCT column1, column2, column3
FROM table1;

which would return the unique (or DISTINCT) rows across all three columns. Note: using DISTINCT, particularly in aggregations, can slow your queries down quite a bit.

Questions

  1. Use DISTINCT to test if there are any accounts associated with more than one region.

The below two queries have the same number of resulting rows (351), so we know that every account is associated with only one region. If each account was associated with more than one region, the first query should have returned more rows than the second query.

SELECT a.id as "account id", r.id as "region id", 
a.name as "account name", r.name as "region name"
FROM accounts a
JOIN sales_reps s
ON s.id = a.sales_rep_id
JOIN region r
ON r.id = s.region_id;

and

SELECT DISTINCT id, name
FROM accounts;
  1. Have any sales reps worked on more than one account?

Actually all of the sales reps have worked on more than one account. The fewest number of accounts any sales rep works on is 3. There are 50 sales reps, and they all have more than one account. Using DISTINCT in the second query assures that all of the sales reps are accounted for in the first query.

SELECT s.id, s.name, COUNT(*) num_accounts
FROM accounts a
JOIN sales_reps s
ON s.id = a.sales_rep_id
GROUP BY s.id, s.name
ORDER BY num_accounts;
SELECT DISTINCT id, name
FROM sales_reps;

HAVING

HAVING is the “clean” way to filter a query that has been aggregated, but this is also commonly done using a subquery. Essentially, any time you want to perform a WHERE on an element of your query that was created by an aggregate, you need to use HAVING instead. HAVING appears after the GROUP BY clause but before the ORDER BY clause.

Questions

  1. How many of the sales reps have more than 5 accounts that they manage?
SELECT s.id, s.name, COUNT(*) num_accounts
FROM accounts a
JOIN sales_reps s
ON s.id = a.sales_rep_id
GROUP BY s.id, s.name
HAVING COUNT(*) > 5
ORDER BY num_accounts;
  1. How many accounts have more than 20 orders?
SELECT a.id, a.name, COUNT(*) num_orders
FROM accounts a
JOIN orders o
ON a.id = o.account_id
GROUP BY a.id, a.name
HAVING COUNT(*) > 20
ORDER BY num_orders;
  1. Which account has the most orders?
SELECT a.id, a.name, COUNT(*) num_orders
FROM accounts a
JOIN orders o
ON a.id = o.account_id
GROUP BY a.id, a.name
ORDER BY num_orders DESC
LIMIT 1;
  1. How many accounts spent more than 30,000 usd total across all orders?
SELECT a.id, a.name, SUM(o.total_amt_usd) total_spent
FROM accounts a
JOIN orders o
ON a.id = o.account_id
GROUP BY a.id, a.name
HAVING SUM(o.total_amt_usd) > 30000
ORDER BY total_spent;
  1. How many accounts spent less than 1,000 usd total across all orders?
SELECT a.id, a.name, SUM(o.total_amt_usd) total_spent
FROM accounts a
JOIN orders o
ON a.id = o.account_id
GROUP BY a.id, a.name
HAVING SUM(o.total_amt_usd) < 1000
ORDER BY total_spent;
  1. Which account has spent the most with us?
SELECT a.id, a.name, SUM(o.total_amt_usd) total_spent
FROM accounts a
JOIN orders o
ON a.id = o.account_id
GROUP BY a.id, a.name
ORDER BY total_spent DESC
LIMIT 1;
  1. Which account has spent the least with us?
SELECT a.id, a.name, SUM(o.total_amt_usd) total_spent
FROM accounts a
JOIN orders o
ON a.id = o.account_id
GROUP BY a.id, a.name
ORDER BY total_spent
LIMIT 1;
  1. Which accounts used facebook as a channel to contact customers more than 6 times?
SELECT a.id, a.name, w.channel, COUNT(*) use_of_channel
FROM accounts a
JOIN web_events w
ON a.id = w.account_id
GROUP BY a.id, a.name, w.channel
HAVING COUNT(*) > 6 AND w.channel = 'facebook'
ORDER BY use_of_channel;
  1. Which account used facebook most as a channel?
SELECT a.id, a.name, w.channel, COUNT(*) use_of_channel
FROM accounts a
JOIN web_events w
ON a.id = w.account_id
WHERE w.channel = 'facebook'
GROUP BY a.id, a.name, w.channel
ORDER BY use_of_channel DESC
LIMIT 1;
  1. Which channel was most frequently used by most accounts?
SELECT a.id, a.name, w.channel, COUNT(*) use_of_channel
FROM accounts a
JOIN web_events w
ON a.id = w.account_id
GROUP BY a.id, a.name, w.channel
ORDER BY use_of_channel DESC
LIMIT 10;

DATE Functions

GROUPing BY a date column is not usually very useful in SQL, as these columns tend to have transaction data down to a second. Keeping date information at such a granular data is both a blessing and a curse, as it gives really precise information (a blessing), but it makes grouping information together directly difficult (a curse). Dates are stored in year, month, day, hour, minute, second, which helps us in truncating. There are a number of built in SQL functions that are aimed at helping us improve our experience in working with dates: In SQL dates are stored in the format YYYY-MM-DD with the time in hours, minutes and seconds at the end. DATE_TRUNC allows you to truncate your date to a particular part of your date-time column. Common trunctions are day, month, and year. DATE_PART can be useful for pulling a specific portion of a date, but notice pulling month or day of the week (dow) means that you are no longer keeping the years in order. Rather you are grouping for certain components regardless of which year they belonged in. Note: DOW returns a value from 0-6 where 0 is Sunday and 6 is Saturday.

Questions

  1. Find the sales in terms of total dollars for all orders in each year, ordered from greatest to least. Do you notice any trends in the yearly sales totals?
SELECT DATE_PART('year', occurred_at) ord_year,  SUM(total_amt_usd) total_spent
FROM orders
GROUP BY 1
ORDER BY 2 DESC;

When we look at the yearly totals, you might notice that 2013 and 2017 have much smaller totals than all other years. If we look further at the monthly data, we see that for 2013 and 2017 there is only one month of sales for each of these years (12 for 2013 and 1 for 2017). Therefore, neither of these are evenly represented. Sales have been increasing year over year, with 2016 being the largest sales to date. At this rate, we might expect 2017 to have the largest sales.

  1. Which month did Parch & Posey have the greatest sales in terms of total dollars? Are all months evenly represented by the dataset?

In order for this to be ‘fair’, we should remove the sales from 2013 and 2017. For the same reasons as discussed above.

SELECT DATE_PART('month', occurred_at) ord_month, SUM(total_amt_usd) total_spent
FROM orders
WHERE occurred_at BETWEEN '2014-01-01' AND '2017-01-01'
GROUP BY 1
ORDER BY 2 DESC;

The greatest sales amounts occur in December (12).

  1. Which year did Parch & Posey have the greatest sales in terms of total number of orders? Are all years evenly represented by the dataset?
SELECT DATE_PART('year', occurred_at) ord_year,  COUNT(*) total_sales
FROM orders
GROUP BY 1
ORDER BY 2 DESC;

Again, 2016 by far has the most amount of orders, but again 2013 and 2017 are not evenly represented to the other years in the dataset.

  1. Which month did Parch & Posey have the greatest sales in terms of total number of orders? Are all months evenly represented by the dataset?
SELECT DATE_PART('month', occurred_at) ord_month, COUNT(*) total_sales
FROM orders
WHERE occurred_at BETWEEN '2014-01-01' AND '2017-01-01'
GROUP BY 1
ORDER BY 2 DESC;

December still has the most sales, but interestingly, November has the second most sales (but not the most dollar sales. To make a fair comparison from one month to another 2017 and 2013 data were removed.

  1. In which month of which year did Walmart spend the most on gloss paper in terms of dollars?
SELECT DATE_TRUNC('month', o.occurred_at) ord_date, SUM(o.gloss_amt_usd) tot_spent
FROM orders o 
JOIN accounts a
ON a.id = o.account_id
WHERE a.name = 'Walmart'
GROUP BY 1
ORDER BY 2 DESC
LIMIT 1;

May 2016 was when Walmart spent the most on gloss paper.

CASE

The CASE statement always goes in the SELECT clause. CASE must include the following components: WHEN, THEN, and END. ELSE is an optional component to catch cases that didn’t meet any of the other previous CASE conditions. You can make any conditional statement using any conditional operator (like WHERE) between WHEN and THEN. This includes stringing together multiple conditional statements using AND and OR. You can include multiple WHEN statements, as well as an ELSE statement again, to deal with any unaddressed conditions. ### Example In a quiz question in the previous Basic SQL lesson, you saw this question: 1. Create a column that divides the standard_amt_usd by the standard_qty to find the unit price for standard paper for each order. Limit the results to the first 10 orders, and include the id and account_id fields. NOTE - you will be thrown an error with the correct solution to this question. This is for a division by zero. You will learn how to get a solution without an error to this query when you learn about CASE statements in a later section. Let’s see how we can use the CASE statement to get around this error.

SELECT id, account_id, standard_amt_usd/standard_qty AS unit_price
FROM orders
LIMIT 10;

Now, let’s use a CASE statement. This way any time the standard_qty is zero, we will return 0, and otherwise we will return the unit_price.

SELECT account_id, CASE WHEN standard_qty = 0 OR standard_qty IS NULL THEN 0
                        ELSE standard_amt_usd/standard_qty END AS unit_price
FROM orders
LIMIT 10;

Now the first part of the statement will catch any of those division by zero values that were causing the error, and the other components will compute the division as necessary. You will notice, we essentially charge all of our accounts 4.99 for standard paper. It makes sense this doesn’t fluctuate, and it is more accurate than adding 1 in the denominator like our quick fix might have been in the earlier lesson.

Questions

  1. We would like to understand 3 different levels of customers based on the amount associated with their purchases. The top branch includes anyone with a Lifetime Value (total sales of all orders) greater than 200,000 usd. The second branch is between 200,000 and 100,000 usd. The lowest branch is anyone under 100,000 usd. Provide a table that includes the level associated with each account. You should provide the account name, the total sales of all orders for the customer, and the level. Order with the top spending customers listed first.
SELECT a.name, SUM(total_amt_usd) total_spent, 
     CASE WHEN SUM(total_amt_usd) > 200000 THEN 'top'
     WHEN  SUM(total_amt_usd) > 100000 THEN 'middle'
     ELSE 'low' END AS customer_level
FROM orders o
JOIN accounts a
ON o.account_id = a.id 
GROUP BY a.name
ORDER BY 2 DESC;
  1. We would now like to perform a similar calculation to the first, but we want to obtain the total amount spent by customers only in 2016 and 2017. Keep the same levels as in the previous question. Order with the top spending customers listed first.
SELECT a.name, SUM(total_amt_usd) total_spent, 
     CASE WHEN SUM(total_amt_usd) > 200000 THEN 'top'
     WHEN  SUM(total_amt_usd) > 100000 THEN 'middle'
     ELSE 'low' END AS customer_level
FROM orders o
JOIN accounts a
ON o.account_id = a.id
WHERE occurred_at > '2015-12-31' 
GROUP BY 1
ORDER BY 2 DESC;
  1. We would like to identify top performing sales reps, which are sales reps associated with more than 200 orders. Create a table with the sales rep name, the total number of orders, and a column with top or not depending on if they have more than 200 orders. Place the top sales people first in your final table.
SELECT s.name, COUNT(*) num_ords,
     CASE WHEN COUNT(*) > 200 THEN 'top'
     ELSE 'not' END AS sales_rep_level
FROM orders o
JOIN accounts a
ON o.account_id = a.id 
JOIN sales_reps s
ON s.id = a.sales_rep_id
GROUP BY s.name
ORDER BY 2 DESC;

It is worth mentioning that this assumes each name is unique - which has been done a few times. We otherwise would want to break by the name and the id of the table.

  1. The previous didn’t account for the middle, nor the dollar amount associated with the sales. Management decides they want to see these characteristics represented as well. We would like to identify top performing sales reps, which are sales reps associated with more than 200 orders or more than 750000 in total sales. The middle group has any rep with more than 150 orders or 500000 in sales. Create a table with the sales rep name, the total number of orders, total sales across all orders, and a column with top, middle, or low depending on this criteria. Place the top sales people based on dollar amount of sales first in your final table. You might see a few upset sales people by this criteria!
SELECT s.name, COUNT(*), SUM(o.total_amt_usd) total_spent, 
     CASE WHEN COUNT(*) > 200 OR SUM(o.total_amt_usd) > 750000 THEN 'top'
     WHEN COUNT(*) > 150 OR SUM(o.total_amt_usd) > 500000 THEN 'middle'
     ELSE 'low' END AS sales_rep_level
FROM orders o
JOIN accounts a
ON o.account_id = a.id 
JOIN sales_reps s
ON s.id = a.sales_rep_id
GROUP BY s.name
ORDER BY 3 DESC;
LS0tCnRpdGxlOiAiTm90ZXMgb24gU1FMIgpvdXRwdXQ6IGh0bWxfbm90ZWJvb2sKLS0tCiMjIEVudGl0eSByZWxhdGlvbnNoaXAgZGlhZ3JhbXMKVGhlICJjcm93J3MgZm9vdCIgdGhhdCBjb25uZWN0cyB0aGUgdGFibGVzIHRvZ2V0aGVyIHNob3dzIHVzIGhvdyB0aGUgY29sdW1ucyBpbiBvbmUgdGFibGUgcmVsYXRlIHRvIHRoZSBjb2x1bW5zIGluIGFub3RoZXIgdGFibGUuIAohW10oaW1hZ2VzL0VSRC5wbmcpCiMjIFR5cGVzIE9mIFN0YXRlbWVudHMKIEEgZmV3IHN0YXRlbWVudHMgaW5jbHVkZToKKiBDUkVBVEUgVEFCTEUgaXMgYSBzdGF0ZW1lbnQgdGhhdCBjcmVhdGVzIGEgbmV3IHRhYmxlIGluIGEgZGF0YWJhc2UuCiogRFJPUCBUQUJMRSBpcyBhIHN0YXRlbWVudCB0aGF0IHJlbW92ZXMgYSB0YWJsZSBpbiBhIGRhdGFiYXNlLgoqIFNFTEVDVCBhbGxvd3MgeW91IHRvIHJlYWQgZGF0YSBhbmQgZGlzcGxheSBpdC4gVGhpcyBpcyBjYWxsZWQgYSBxdWVyeS4KCiMjIFNFTEVDVCAmIEZST00gU3RhdGVtZW50cwpIZXJlIHlvdSB3ZXJlIGludHJvZHVjZWQgdG8gdHdvIHN0YXRlbWVudHMgdGhhdCB3aWxsIGJlIHVzZWQgaW4gZXZlcnkgcXVlcnkgeW91IHdyaXRlCiogU0VMRUNUIC0gd2hhdCBjb2x1bW5zIHlvdSB3YW50IGJhY2suCiogRlJPTSAtIHdoYXQgdGFibGUgeW91IGFyZSBxdWVyeWluZyBmcm9tLiAKYGBge3NxbH0KU0VMRUNUIGlkLCBhY2NvdW50X2lkLCBvY2N1cnJlZF9hdApGUk9NIG9yZGVyczsKYGBgCgojIyBMSU1JVCBTdGF0ZW1lbnQKIFRoZSBMSU1JVCBzdGF0ZW1lbnQgaXMgdXNlZnVsIHdoZW4geW91IHdhbnQgdG8gc2VlIGp1c3QgdGhlIGZpcnN0IGZldyByb3dzIG9mIGEgdGFibGUuIFRoaXMgY2FuIGJlIG11Y2ggZmFzdGVyIGZvciBsb2FkaW5nIHRoYW4gaWYgd2UgbG9hZCB0aGUgZW50aXJlIGRhdGFzZXQuClRoZSBMSU1JVCBjb21tYW5kIGlzIGFsd2F5cyB0aGUgdmVyeSBsYXN0IHBhcnQgb2YgYSBxdWVyeS4gQW4gZXhhbXBsZSBvZiBzaG93aW5nIGp1c3QgdGhlIGZpcnN0IDEwIHJvd3Mgb2YgdGhlIG9yZGVycyB0YWJsZSB3aXRoIGFsbCBvZiB0aGUgY29sdW1ucyBtaWdodCBsb29rIGxpa2UgdGhlIGZvbGxvd2luZzoKYGBge3NxbH0KU0VMRUNUICoKRlJPTSBvcmRlcnMKTElNSVQgMTAKYGBgCgojIyBPUkRFUiBCWSBTdGF0ZW1lbnQKKiBUaGUgT1JERVIgQlkgc3RhdGVtZW50IGFsbG93cyB1cyB0byBvcmRlciBvdXIgdGFibGUgYnkgYW55IHJvdy4gCiogUmVtZW1iZXIgREVTQyBjYW4gYmUgYWRkZWQgYWZ0ZXIgdGhlIGNvbHVtbiBpbiB5b3VyIE9SREVSIEJZIHN0YXRlbWVudCB0byBzb3J0IGluIGRlc2NlbmRpbmcgb3JkZXIsIGFzIHRoZSBkZWZhdWx0IGlzIHRvIHNvcnQgaW4gYXNjZW5kaW5nIG9yZGVyLgpgYGB7c3FsfQpTRUxFQ1QgaWQsIG9jY3VycmVkX2F0LCB0b3RhbF9hbXRfdXNkCkZST00gb3JkZXJzCk9SREVSIEJZIG9jY3VycmVkX2F0CkxJTUlUIDEwOwpgYGAKCldlIGNhbiBPUkRFUiBCWSBtb3JlIHRoYW4gb25lIGNvbHVtbiBhdCBhIHRpbWUuIFRoZSBzdGF0ZW1lbnQgc29ydHMgYWNjb3JkaW5nIHRvIGNvbHVtbnMgbGlzdGVkIGZyb20gbGVmdCBmaXJzdCBhbmQgdGhvc2UgbGlzdGVkIG9uIHRoZSByaWdodCBhZnRlciB0aGF0LgpgYGB7c3FsfQpTRUxFQ1QgKgpGUk9NIG9yZGVycwpPUkRFUiBCWSBvY2N1cnJlZF9hdCwgdG90YWxfYW10X3VzZApMSU1JVCAxMDsKYGBgCgojIyBXSEVSRSBTdGF0ZW1lbnRzClVzaW5nIHRoZSAqKldIRVJFKiogc3RhdGVtZW50LCB3ZSBjYW4gc3Vic2V0IG91dCB0YWJsZXMgYmFzZWQgb24gY29uZGl0aW9ucyB0aGF0IG11c3QgYmUgbWV0LgpgYGB7c3FsfQpTRUxFQ1QgKgpGUk9NIG9yZGVycwpXSEVSRSBnbG9zc19hbXRfdXNkID49IDEwMDAKTElNSVQgNTsKYGBgCgpUaGUgV0hFUkUgc3RhdGVtZW50IGNhbiBhbHNvIGJlIHVzZWQgd2l0aCBub24tbnVtZXJpY2FsIGRhdGEuIFdlIGNhbiB1c2UgdGhlID0gYW5kICE9IG9wZXJhdG9ycyBoZXJlLiBZb3UgYWxzbyBuZWVkIHRvIGJlIHN1cmUgdG8gdXNlIHNpbmdsZSBxdW90ZXMKYGBge3NxbH0KU0VMRUNUIG5hbWUsIHdlYnNpdGUsIHByaW1hcnlfcG9jCkZST00gYWNjb3VudHMKV0hFUkUgbmFtZSA9ICdFeHhvbiBNb2JpbCc7CmBgYAoKIyMgQXJpdGhtZXRpYyBPcGVyYXRvcnMKIyMjIERlcml2ZWQgQ29sdW1ucwpDcmVhdGluZyBhIG5ldyBjb2x1bW4gdGhhdCBpcyBhIGNvbWJpbmF0aW9uIG9mIGV4aXN0aW5nIGNvbHVtbnMgaXMga25vd24gYXMgYSBkZXJpdmVkIGNvbHVtbi4KIyMjIE9yZGVyIG9mIE9wZXJhdGlvbnMKUmVtZW1iZXIgQk9NREFTIGZyb20gbWF0aHMgY2xhc3M/IFRoZSBzYW1lIG9yZGVyIG9mIG9wZXJhdGlvbnMgYXBwbHkgd2hlbiB1c2luZyBhcml0aG1ldGljIG9wZXJhdG9ycyBpbiBTUUwuCmBgYHtzcWx9ClNFTEVDVCBpZCwgYWNjb3VudF9pZCwgCiAgICAgICBwb3N0ZXJfYW10X3VzZC8oc3RhbmRhcmRfYW10X3VzZCArIGdsb3NzX2FtdF91c2QgKyBwb3N0ZXJfYW10X3VzZCkgQVMgcG9zdF9wZXIKRlJPTSBvcmRlcnM7CmBgYAojIyMgSW50cm9kdWN0aW9uIHRvIExvZ2ljYWwgT3BlcmF0b3JzCgoqKkxJS0UqKiAtIFRoaXMgYWxsb3dzIHlvdSB0byBwZXJmb3JtIG9wZXJhdGlvbnMgc2ltaWxhciB0byB1c2luZyBXSEVSRSBhbmQgPSwgYnV0IGZvciBjYXNlcyB3aGVuIHlvdSBtaWdodCBub3Qga25vdyBleGFjdGx5IHdoYXQgeW91IGFyZSBsb29raW5nIGZvci4gIFRoZSBMSUtFIG9wZXJhdG9yIGlzIGZyZXF1ZW50bHkgdXNlZCB3aXRoICUuIFRoZSAlIHRlbGxzIHVzIHRoYXQgd2UgbWlnaHQgd2FudCBhbnkgbnVtYmVyIG9mIGNoYXJhY3RlcnMgbGVhZGluZyB1cCB0byBhIHBhcnRpY3VsYXIgc2V0IG9mIGNoYXJhY3RlcnMgb3IgZm9sbG93aW5nIGEgY2VydGFpbiBzZXQgb2YgY2hhcmFjdGVycy4KYGBge3NxbH0KLyogc2VsZWN0IGFsbCB0aGUgY29tcGFuaWVzIHdob3NlIG5hbWVzIHN0YXJ0IHdpdGggJ0MnICovClNFTEVDVCBuYW1lIApGUk9NIGFjY291bnRzCldIRVJFIG5hbWUgbGlrZSAnQyUnCgovKiBzZWxlY3QgQWxsIGNvbXBhbmllcyB3aG9zZSBuYW1lcyBjb250YWluIHRoZSBzdHJpbmcgJ29uZScgc29tZXdoZXJlIGluIHRoZSBuYW1lLiAqLwpTRUxFQ1QgbmFtZQpGUk9NIGFjY291bnRzCldIRVJFIG5hbWUgTElLRSAnJW9uZSUnOwpgYGAKCioqSU4qKiAtIFRoaXMgYWxsb3dzIHlvdSB0byBwZXJmb3JtIG9wZXJhdGlvbnMgc2ltaWxhciB0byB1c2luZyBXSEVSRSBhbmQgPSwgYnV0IGZvciBtb3JlIHRoYW4gb25lIGNvbmRpdGlvbi4gIFdlIGNhbiBjaGVjayBvbmUsIHR3byBvciBtYW55IGNvbHVtbiB2YWx1ZXMgZm9yIHdoaWNoIHdlIHdhbnQgdG8gcHVsbCBkYXRhLCBidXQgYWxsIHdpdGhpbiB0aGUgc2FtZSBxdWVyeS4gClRJUCAtIEluIG1vc3QgU1FMIGVudmlyb25tZW50cywgeW91IGNhbiB1c2Ugc2luZ2xlIG9yIGRvdWJsZSBxdW90YXRpb24gbWFya3MgLSBhbmQgeW91IG1heSBORUVEIHRvIHVzZSBkb3VibGUgcXVvdGF0aW9uIG1hcmtzIGlmIHlvdSBoYXZlIGFuIGFwb3N0cm9waGUgd2l0aGluIHRoZSB0ZXh0IHlvdSBhcmUgYXR0ZW1wdGluZyB0byBwdWxsLgpgYGB7c3FsfQpTRUxFQ1QgbmFtZSwgcHJpbWFyeV9wb2MsIHNhbGVzX3JlcF9pZApGUk9NIGFjY291bnRzCldIRVJFIG5hbWUgSU4gKCdXYWxtYXJ0JywgJ1RhcmdldCcsICdOb3Jkc3Ryb20nKTsKYGBgCioqTk9UKiogLSBUaGlzIGlzIHVzZWQgd2l0aCBJTiBhbmQgTElLRSB0byBzZWxlY3QgYWxsIG9mIHRoZSByb3dzIE5PVCBMSUtFIG9yIE5PVCBJTiBhIGNlcnRhaW4gY29uZGl0aW9uLgpgYGB7c3FsfQovKiBVc2UgdGhlIGFjY291bnRzIHRhYmxlIHRvIGZpbmQgdGhlIGFjY291bnQgbmFtZSwgcHJpbWFyeSBwb2MsIGFuZCBzYWxlcyByZXAgaWQgZm9yIGFsbCBzdG9yZXMgZXhjZXB0IFdhbG1hcnQsIFRhcmdldCwgYW5kIE5vcmRzdHJvbSAqLwpTRUxFQ1QgbmFtZSwgcHJpbWFyeV9wb2MsIHNhbGVzX3JlcF9pZApGUk9NIGFjY291bnRzCldIRVJFIG5hbWUgTk9UIElOICgnV2FsbWFydCcsICdUYXJnZXQnLCAnTm9yZHN0cm9tJyk7CmBgYAoKKipBTkQqKiAmICoqQkVUV0VFTioqIC0gVGhlc2UgYWxsb3cgeW91IHRvIGNvbWJpbmUgb3BlcmF0aW9ucyB3aGVyZSBhbGwgY29tYmluZWQgY29uZGl0aW9ucyBtdXN0IGJlIHRydWUuCiogVGhlIEFORCBvcGVyYXRvciBpcyB1c2VkIHdpdGhpbiBhIFdIRVJFIHN0YXRlbWVudCB0byBjb25zaWRlciBtb3JlIHRoYW4gb25lIGxvZ2ljYWwgY2xhdXNlIGF0IGEgdGltZS4gRWFjaCB0aW1lIHlvdSBsaW5rIGEgbmV3IHN0YXRlbWVudCB3aXRoIGFuIEFORCwgeW91IHdpbGwgbmVlZCB0byBzcGVjaWZ5IHRoZSBjb2x1bW4geW91IGFyZSBpbnRlcmVzdGVkIGluIGxvb2tpbmcgYXQuCiogU29tZXRpbWVzIHdlIGNhbiBtYWtlIGEgY2xlYW5lciBzdGF0ZW1lbnQgdXNpbmcgQkVUV0VFTiB0aGFuIHdlIGNhbiB1c2luZyBBTkQuIFBhcnRpY3VsYXJseSB0aGlzIGlzIHRydWUgd2hlbiB3ZSBhcmUgdXNpbmcgdGhlIHNhbWUgY29sdW1uIGZvciBkaWZmZXJlbnQgcGFydHMgb2Ygb3VyIEFORCBzdGF0ZW1lbnQuIGUuZy46CmBgYHtzcWx9Ci8qSW5zdGVhZCBvZiB3cml0aW5nIDogV0hFUkUgY29sdW1uID49IDYgQU5EIGNvbHVtbiA8PSAxMCB3ZSBjYW4gaW5zdGVhZCB3cml0ZSovIApXSEVSRSBjb2x1bW4gQkVUV0VFTiA2IEFORCAxMApgYGAKCmBgYHtzcWx9Ci8qIFdyaXRlIGEgcXVlcnkgdGhhdCByZXR1cm5zIGFsbCB0aGUgb3JkZXJzIHdoZXJlIHRoZSBzdGFuZGFyZF9xdHkgaXMgb3ZlciAxMDAwLCB0aGUgcG9zdGVyX3F0eSBpcyAwLCBhbmQgdGhlIGdsb3NzX3F0eSBpcyAwICovClNFTEVDVCAqCkZST00gb3JkZXJzCldIRVJFIHN0YW5kYXJkX3F0eSA+IDEwMDAgQU5EIHBvc3Rlcl9xdHkgPSAwIEFORCBnbG9zc19xdHkgPSAwOwoKLyogVXNpbmcgdGhlIGFjY291bnRzIHRhYmxlIGZpbmQgYWxsIHRoZSBjb21wYW5pZXMgd2hvc2UgbmFtZXMgZG8gbm90IHN0YXJ0IHdpdGggJ0MnIGFuZCBlbmQgd2l0aCAncycgKi8KU0VMRUNUIG5hbWUKRlJPTSBhY2NvdW50cwpXSEVSRSBuYW1lIE5PVCBMSUtFICdDJScgQU5EIG5hbWUgTElLRSAnJXMnOwoKLyogVXNlIHRoZSB3ZWJfZXZlbnRzIHRhYmxlIHRvIGZpbmQgYWxsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBpbmRpdmlkdWFscyB3aG8gd2VyZSBjb250YWN0ZWQgdmlhIG9yZ2FuaWMgb3IgYWR3b3JkcyBhbmQgc3RhcnRlZCB0aGVpciBhY2NvdW50IGF0IGFueSBwb2ludCBpbiAyMDE2IHNvcnRlZCBmcm9tIG5ld2VzdCB0byBvbGRlc3QgKi8gClNFTEVDVCAqCkZST00gd2ViX2V2ZW50cwpXSEVSRSBjaGFubmVsIElOICgnb3JnYW5pYycsICdhZHdvcmRzJykgQU5EIG9jY3VycmVkX2F0IEJFVFdFRU4gJzIwMTYtMDEtMDEnIEFORCAnMjAxNy0wMS0wMScKT1JERVIgQlkgb2NjdXJyZWRfYXQgREVTQzsKLyogIFlvdSB3aWxsIG5vdGljZSB0aGF0IHVzaW5nIEJFVFdFRU4gaXMgdHJpY2t5IGZvciBkYXRlcyEgV2hpbGUgQkVUV0VFTiBpcyBnZW5lcmFsbHkgaW5jbHVzaXZlIG9mIGVuZHBvaW50cywgaXQgYXNzdW1lcyB0aGUgdGltZSBpcyBhdCAwMDowMDowMCAoaS5lLiBtaWRuaWdodCkgZm9yIGRhdGVzLiBUaGlzIGlzIHRoZSByZWFzb24gd2h5IHdlIHNldCB0aGUgcmlnaHQtc2lkZSBlbmRwb2ludCBvZiB0aGUgcGVyaW9kIGF0ICcyMDE3LTAxLTAxJy4gKi8KYGBgCgoqKk9SKiogLSBUaGlzIGFsbG93IHlvdSB0byBjb21iaW5lIG9wZXJhdGlvbnMgd2hlcmUgYXQgbGVhc3Qgb25lIG9mIHRoZSBjb21iaW5lZCBjb25kaXRpb25zIG11c3QgYmUgdHJ1ZS4KU2ltaWxhciB0byB0aGUgQU5EIG9wZXJhdG9yLCB0aGUgT1Igb3BlcmF0b3IgY2FuIGNvbWJpbmUgbXVsdGlwbGUgc3RhdGVtZW50cy4gRWFjaCB0aW1lIHlvdSBsaW5rIGEgbmV3IHN0YXRlbWVudCB3aXRoIGFuIE9SLCB5b3Ugd2lsbCBuZWVkIHRvIHNwZWNpZnkgdGhlIGNvbHVtbiB5b3UgYXJlIGludGVyZXN0ZWQgaW4gbG9va2luZyBhdC4KYGBge3NxbH0KLyogRmluZCBsaXN0IG9mIG9yZGVycyBpZHMgd2hlcmUgZWl0aGVyIGdsb3NzX3F0eSBvciBwb3N0ZXJfcXR5IGlzIGdyZWF0ZXIgdGhhbiA0MDAwLiBPbmx5IGluY2x1ZGUgdGhlIGlkIGZpZWxkIGluIHRoZSByZXN1bHRpbmcgdGFibGUuICovClNFTEVDVCBpZApGUk9NIG9yZGVycwpXSEVSRSBnbG9zc19xdHkgPiA0MDAwIE9SIHBvc3Rlcl9xdHkgPiA0MDAwOwoKLyogV3JpdGUgYSBxdWVyeSB0aGF0IHJldHVybnMgYSBsaXN0IG9mIG9yZGVycyB3aGVyZSB0aGUgc3RhbmRhcmRfcXR5IGlzIHplcm8gYW5kIGVpdGhlciB0aGUgZ2xvc3NfcXR5IG9yIHBvc3Rlcl9xdHkgaXMgb3ZlciAxMDAwLiAqLwpTRUxFQ1QgKgpGUk9NIG9yZGVycwpXSEVSRSBzdGFuZGFyZF9xdHkgPSAwIEFORCAoZ2xvc3NfcXR5ID4gMTAwMCBPUiBwb3N0ZXJfcXR5ID4gMTAwMCk7CgovKiBGaW5kIGFsbCB0aGUgY29tcGFueSBuYW1lcyB0aGF0IHN0YXJ0IHdpdGggYSAnQycgb3IgJ1cnLCBhbmQgdGhlIHByaW1hcnkgY29udGFjdCBjb250YWlucyAnYW5hJyBvciAnQW5hJywgYnV0IGl0IGRvZXNuJ3QgY29udGFpbiAnZWFuYScqLwpTRUxFQ1QgKgpGUk9NIGFjY291bnRzCldIRVJFIChuYW1lIExJS0UgJ0MlJyBPUiBuYW1lIExJS0UgJ1clJykgCiAgICAgICAgICAgQU5EICgocHJpbWFyeV9wb2MgTElLRSAnJWFuYSUnIE9SIHByaW1hcnlfcG9jIExJS0UgJyVBbmElJykgCiAgICAgICAgICAgQU5EIHByaW1hcnlfcG9jIE5PVCBMSUtFICclZWFuYSUnKTsKYGBgCgojIyBTdW1tYXJ5IG9mIGJhc2ljIGNvbWFuZHM6CiFbXShpbWFnZXMvc3VtbWFyeV9iYXNpY19jb21tYW5kcy5wbmcpCgoqKioKCiMgSk9JTlMKIyMgSW5uZXIgam9pbnMgCk9ubHkgcmV0dXJuIHJvd3MgdGhhdCBhcHBlYXIgaW4gKipib3RoIHRhYmxlcyoqCiFbXShpbWFnZXMvRVJELnBuZykgCmUuZy4gVXNpbmcgRVJEOiBQdWxsIHN0YW5kYXJkX3F0eSwgZ2xvc3NfcXR5LCBhbmQgcG9zdGVyX3F0eSBmcm9tIHRoZSBvcmRlcnMgdGFibGUsIGFuZCB0aGUgd2Vic2l0ZSBhbmQgdGhlIHByaW1hcnlfcG9jIGZyb20gdGhlIGFjY291bnRzIHRhYmxlLgpgYGB7c3FsfQpTRUxFQ1Qgb3JkZXJzLnN0YW5kYXJkX3F0eSwgb3JkZXJzLmdsb3NzX3F0eSwgCiAgICAgICBvcmRlcnMucG9zdGVyX3F0eSwgIGFjY291bnRzLndlYnNpdGUsIAogICAgICAgYWNjb3VudHMucHJpbWFyeV9wb2MKRlJPTSBvcmRlcnMKSk9JTiBhY2NvdW50cwpPTiBvcmRlcnMuYWNjb3VudF9pZCA9IGFjY291bnRzLmlkCmBgYApOb3RpY2UgdGhhdCB3ZSBuZWVkIHRvIHNwZWNpZnkgZXZlcnkgdGFibGUgYSBjb2x1bW4gY29tZXMgZnJvbSBpbiB0aGUgU0VMRUNUIHN0YXRlbWVudC4KIyMjIEFMSUFTCldoZW4gd2UgSk9JTiB0YWJsZXMgdG9nZXRoZXIsIGl0IGlzIG5pY2UgdG8gZ2l2ZSBlYWNoIHRhYmxlIGFuIGFsaWFzLiBGcmVxdWVudGx5IGFuIGFsaWFzIGlzIGp1c3QgdGhlIGZpcnN0IGxldHRlciBvZiB0aGUgdGFibGUgbmFtZS4gCmBgYHtzcWx9CkZST00gdGFibGVuYW1lIHQxIApKT0lOIHRhYmxlbmFtZTIgdDIKYGBgCgojIyMgUXVlc3Rpb25zCjEuIFByb3ZpZGUgYSB0YWJsZSBmb3IgYWxsICp3ZWJfZXZlbnRzKiBhc3NvY2lhdGVkIHdpdGggYWNjb3VudCBuYW1lIG9mICpXYWxtYXJ0Ki4gVGhlcmUgc2hvdWxkIGJlIHRocmVlIGNvbHVtbnMuIEJlIHN1cmUgdG8gaW5jbHVkZSB0aGUgcHJpbWFyeV9wb2MsIHRpbWUgb2YgdGhlIGV2ZW50LCBhbmQgdGhlIGNoYW5uZWwgZm9yIGVhY2ggZXZlbnQuIEFkZGl0aW9uYWxseSwgeW91IG1pZ2h0IGNob29zZSB0byBhZGQgYSBmb3VydGggY29sdW1uIHRvIGFzc3VyZSBvbmx5IFdhbG1hcnQgZXZlbnRzIHdlcmUgY2hvc2VuLiAKYGBge3NxbH0KU0VMRUNUIGEucHJpbWFyeV9wb2MsIHcub2NjdXJyZWRfYXQsIHcuY2hhbm5lbCwgYS5uYW1lCkZST00gd2ViX2V2ZW50cyB3CkpPSU4gYWNjb3VudHMgYQpPTiB3LmFjY291bnRfaWQgPSBhLmlkCldIRVJFIGEubmFtZSA9ICdXYWxtYXJ0JzsKYGBgCgoyLiBQcm92aWRlIGEgdGFibGUgdGhhdCBwcm92aWRlcyB0aGUgKnJlZ2lvbiogZm9yIGVhY2ggKnNhbGVzX3JlcCogYWxvbmcgd2l0aCB0aGVpciBhc3NvY2lhdGVkICphY2NvdW50cyouIFlvdXIgZmluYWwgdGFibGUgc2hvdWxkIGluY2x1ZGUgdGhyZWUgY29sdW1uczogdGhlIHJlZ2lvbiAqbmFtZSosIHRoZSBzYWxlcyByZXAgKm5hbWUqLCBhbmQgdGhlIGFjY291bnQgKm5hbWUqLiBTb3J0IHRoZSBhY2NvdW50cyBhbHBoYWJldGljYWxseSAoQS1aKSBhY2NvcmRpbmcgdG8gYWNjb3VudCBuYW1lLiAKYGBge3NxbH0KU0VMRUNUIHIubmFtZSByZWdpb24sIHMubmFtZSByZXAsIGEubmFtZSBhY2NvdW50CkZST00gc2FsZXNfcmVwcyBzCkpPSU4gcmVnaW9uIHIKT04gcy5yZWdpb25faWQgPSByLmlkCkpPSU4gYWNjb3VudHMgYQpPTiBhLnNhbGVzX3JlcF9pZCA9IHMuaWQKT1JERVIgQlkgYS5uYW1lOwpgYGAKCjMuIFByb3ZpZGUgdGhlICpuYW1lKiBmb3IgZWFjaCByZWdpb24gZm9yIGV2ZXJ5ICpvcmRlciosIGFzIHdlbGwgYXMgdGhlIGFjY291bnQgKm5hbWUqIGFuZCB0aGUgKnVuaXQgcHJpY2UqIHRoZXkgcGFpZCAodG90YWxfYW10X3VzZC90b3RhbCkgZm9yIHRoZSBvcmRlci4gWW91ciBmaW5hbCB0YWJsZSBzaG91bGQgaGF2ZSAzIGNvbHVtbnM6ICpyZWdpb24gbmFtZSwgYWNjb3VudCBuYW1lLCBhbmQgdW5pdCBwcmljZSouIEEgZmV3IGFjY291bnRzIGhhdmUgMCBmb3IgKnRvdGFsKiwgc28gSSBkaXZpZGVkIGJ5ICh0b3RhbCArIDAuMDEpIHRvIGFzc3VyZSBub3QgZGl2aWRpbmcgYnkgemVyby4KYGBge3NxbH0KU0VMRUNUIHIubmFtZSByZWdpb24sIGEubmFtZSBhY2NvdW50LCAKICAgICAgIG8udG90YWxfYW10X3VzZC8oby50b3RhbCArIDAuMDEpIHVuaXRfcHJpY2UKRlJPTSByZWdpb24gcgpKT0lOIHNhbGVzX3JlcHMgcwpPTiBzLnJlZ2lvbl9pZCA9IHIuaWQKSk9JTiBhY2NvdW50cyBhCk9OIGEuc2FsZXNfcmVwX2lkID0gcy5pZApKT0lOIG9yZGVycyBvCk9OIG8uYWNjb3VudF9pZCA9IGEuaWQ7CmBgYAoKIyMgT3V0ZXIgam9pbnMKSWYgd2Ugd2FudCB0byBpbmNsdWRlIGRhdGEgdGhhdCBvbmx5IGV4aXN0cyBpbiBvbmUgb2YgdGhlIHRhYmxlcyB3ZSBhbnQgdG8gam9pbiB3ZSBjYW4gdXNlIGFuIG91dGVyIGpvaW4KIVtdKGltYWdlcy90eXBlc19vZl9qb2lucy5wbmcpIApJZiB0aGVyZSBpcyBub3QgbWF0Y2hpbmcgaW5mb3JtYXRpb24gaW4gdGhlIEpPSU5lZCB0YWJsZSwgdGhlbiB5b3Ugd2lsbCBoYXZlIGNvbHVtbnMgd2l0aCBlbXB0eSBjZWxscyB3aXRoIGEgZGF0YSB0eXBlIGNhbGxlZCBOVUxMLgpXcml0aW5nIGEgTEVGVCBqb2luIHN0YXRlbWVudDoKIVtdKGltYWdlcy9sZWZ0X2pvaW4ucG5nKSAKIVtdKGltYWdlcy9sZWZ0X2pvaW5fcHQyLnBuZykgCkNvbXBhcmluZyBMRUZUIGFuZCBSSUdIVCBqb2luOgohW10oaW1hZ2VzL2NvbXBhcmVfbGVmdF9hbmRfcmlnaHRfam9pbi5wbmcpIApMRUZUIGFuZCBSSUdIVCBqb2lucyBhcmUgZXNzZW50aWFsbHkgaW50ZXIgY2hhbmdlYWJsZSBzbyBpdHMgc3RhbmRhcmQgdG8gKmp1c3QgdXNlICBMRUZUIGpvaW5zLioKCiMjIEpPSU5TIGFuZCBmaWx0ZXJpbmcKSWYgeW91IHdhbnRlZCB0byBmaWx0ZXIgeW91ciByZXN1bHRzIHNvIHRoYXQgaXQgb25seSBzaG93ZWQgc2FsZXMgZnJvbSBhIHBhcnRpY3VsYXIgc2FsZXMgcmVwIHlvdSBjb3VsZCB1c2UgV0hFUkUgYnV0IGl04oCZcyAqKmJldHRlciB0byB1c2UgQU5EIGFzIHBhcnQgb2YgdGhlIE9OIHN0YXRlbWVudC4qKgpBIHNpbXBsZSBydWxlIHRvIHJlbWVtYmVyIHRoaXMgaXMgdGhhdCwgd2hlbiB0aGUgZGF0YWJhc2UgZXhlY3V0ZXMgdGhpcyBxdWVyeSwgaXQgZXhlY3V0ZXMgdGhlIGpvaW4gYW5kIGV2ZXJ5dGhpbmcgaW4gdGhlIE9OIGNsYXVzZSBmaXJzdC4gVGhpbmsgb2YgdGhpcyBhcyBidWlsZGluZyB0aGUgbmV3IHJlc3VsdCBzZXQuIFRoYXQgcmVzdWx0IHNldCBpcyB0aGVuIGZpbHRlcmVkIHVzaW5nIHRoZSBXSEVSRSBjbGF1c2UuCmBgYHtzcWx9ClNFTEVDVCBvcmRlcnMuKiwgCmFjY291bnRzLioKRlJPTSBvcmRlcnMKTEVGVCBKT0lOIGFjY291bnRzCk9OIG9yZGVycy5hY2NvdW50X2lkID0gYWNjb3VudHMuaWQKLyogVXNlIEFORCBpbiBzdGVhZCBvZiBXSEVSRSBmb3IgZmlsdGVyaW5nIHRoaXMgTEVGVCBKT0lOICovCkFORCBhY2NvdW50cy5zYWxlc19yZXBfaWQgPSAzMjUgCmBgYAoKIyMjIFF1ZXN0aW9ucwoxLiBQcm92aWRlIGEgdGFibGUgdGhhdCBwcm92aWRlcyB0aGUgcmVnaW9uIGZvciBlYWNoIHNhbGVzX3JlcCBhbG9uZyB3aXRoIHRoZWlyIGFzc29jaWF0ZWQgYWNjb3VudHMuIFRoaXMgdGltZSBvbmx5IGZvciB0aGUgTWlkd2VzdCByZWdpb24uIFlvdXIgZmluYWwgdGFibGUgc2hvdWxkIGluY2x1ZGUgdGhyZWUgY29sdW1uczogdGhlIHJlZ2lvbiBuYW1lLCB0aGUgc2FsZXMgcmVwIG5hbWUsIGFuZCB0aGUgYWNjb3VudCBuYW1lLiBTb3J0IHRoZSBhY2NvdW50cyBhbHBoYWJldGljYWxseSAoQS1aKSBhY2NvcmRpbmcgdG8gYWNjb3VudCBuYW1lLgpgYGB7c3FsfQpTRUxFQ1Qgci5uYW1lIHJlZ2lvbiwgcy5uYW1lIHJlcCwgYS5uYW1lIGFjY291bnQKRlJPTSBzYWxlc19yZXBzIHMKSk9JTiByZWdpb24gcgpPTiBzLnJlZ2lvbl9pZCA9IHIuaWQKSk9JTiBhY2NvdW50cyBhCk9OIGEuc2FsZXNfcmVwX2lkID0gcy5pZApXSEVSRSByLm5hbWUgPSAnTWlkd2VzdCcKT1JERVIgQlkgYS5uYW1lOwpgYGAKCjIuIFByb3ZpZGUgYSB0YWJsZSB0aGF0IHByb3ZpZGVzIHRoZSByZWdpb24gZm9yIGVhY2ggc2FsZXNfcmVwIGFsb25nIHdpdGggdGhlaXIgYXNzb2NpYXRlZCBhY2NvdW50cy4gVGhpcyB0aW1lIG9ubHkgZm9yIGFjY291bnRzIHdoZXJlIHRoZSBzYWxlcyByZXAgaGFzIGEgZmlyc3QgbmFtZSBzdGFydGluZyB3aXRoIFMgYW5kIGluIHRoZSBNaWR3ZXN0IHJlZ2lvbi4gWW91ciBmaW5hbCB0YWJsZSBzaG91bGQgaW5jbHVkZSB0aHJlZSBjb2x1bW5zOiB0aGUgcmVnaW9uIG5hbWUsIHRoZSBzYWxlcyByZXAgbmFtZSwgYW5kIHRoZSBhY2NvdW50IG5hbWUuIFNvcnQgdGhlIGFjY291bnRzIGFscGhhYmV0aWNhbGx5IChBLVopIGFjY29yZGluZyB0byBhY2NvdW50IG5hbWUuIApgYGB7c3FsfQpTRUxFQ1Qgci5uYW1lIHJlZ2lvbiwgcy5uYW1lIHJlcCwgYS5uYW1lIGFjY291bnQKRlJPTSBzYWxlc19yZXBzIHMKSk9JTiByZWdpb24gcgpPTiBzLnJlZ2lvbl9pZCA9IHIuaWQKSk9JTiBhY2NvdW50cyBhCk9OIGEuc2FsZXNfcmVwX2lkID0gcy5pZApXSEVSRSByLm5hbWUgPSAnTWlkd2VzdCcgQU5EIHMubmFtZSBMSUtFICdTJScKT1JERVIgQlkgYS5uYW1lOwpgYGAKCjMuIFByb3ZpZGUgYSB0YWJsZSB0aGF0IHByb3ZpZGVzIHRoZSByZWdpb24gZm9yIGVhY2ggc2FsZXNfcmVwIGFsb25nIHdpdGggdGhlaXIgYXNzb2NpYXRlZCBhY2NvdW50cy4gVGhpcyB0aW1lIG9ubHkgZm9yIGFjY291bnRzIHdoZXJlIHRoZSBzYWxlcyByZXAgaGFzIGEgbGFzdCBuYW1lIHN0YXJ0aW5nIHdpdGggSyBhbmQgaW4gdGhlIE1pZHdlc3QgcmVnaW9uLiBZb3VyIGZpbmFsIHRhYmxlIHNob3VsZCBpbmNsdWRlIHRocmVlIGNvbHVtbnM6IHRoZSByZWdpb24gbmFtZSwgdGhlIHNhbGVzIHJlcCBuYW1lLCBhbmQgdGhlIGFjY291bnQgbmFtZS4gU29ydCB0aGUgYWNjb3VudHMgYWxwaGFiZXRpY2FsbHkgKEEtWikgYWNjb3JkaW5nIHRvIGFjY291bnQgbmFtZS4KYGBge3NxbH0KU0VMRUNUIHIubmFtZSByZWdpb24sIHMubmFtZSByZXAsIGEubmFtZSBhY2NvdW50CkZST00gc2FsZXNfcmVwcyBzCkpPSU4gcmVnaW9uIHIKT04gcy5yZWdpb25faWQgPSByLmlkCkpPSU4gYWNjb3VudHMgYQpPTiBhLnNhbGVzX3JlcF9pZCA9IHMuaWQKV0hFUkUgci5uYW1lID0gJ01pZHdlc3QnIEFORCBzLm5hbWUgTElLRSAnJSBLJScKT1JERVIgQlkgYS5uYW1lOwpgYGAKCjQuIFByb3ZpZGUgdGhlIG5hbWUgZm9yIGVhY2ggcmVnaW9uIGZvciBldmVyeSBvcmRlciwgYXMgd2VsbCBhcyB0aGUgYWNjb3VudCBuYW1lIGFuZCB0aGUgdW5pdCBwcmljZSB0aGV5IHBhaWQgKHRvdGFsX2FtdF91c2QvdG90YWwpIGZvciB0aGUgb3JkZXIuIEhvd2V2ZXIsIHlvdSBzaG91bGQgb25seSBwcm92aWRlIHRoZSByZXN1bHRzIGlmIHRoZSBzdGFuZGFyZCBvcmRlciBxdWFudGl0eSBleGNlZWRzIDEwMC4gWW91ciBmaW5hbCB0YWJsZSBzaG91bGQgaGF2ZSAzIGNvbHVtbnM6IHJlZ2lvbiBuYW1lLCBhY2NvdW50IG5hbWUsIGFuZCB1bml0IHByaWNlLiBJbiBvcmRlciB0byBhdm9pZCBhIGRpdmlzaW9uIGJ5IHplcm8gZXJyb3IsIGFkZGluZyAuMDEgdG8gdGhlIGRlbm9taW5hdG9yIGhlcmUgaXMgaGVscGZ1bCB0b3RhbF9hbXRfdXNkLyh0b3RhbCswLjAxKS4gCmBgYHtzcWx9ClNFTEVDVCByLm5hbWUgcmVnaW9uLCBhLm5hbWUgYWNjb3VudCwgby50b3RhbF9hbXRfdXNkLyhvLnRvdGFsICsgMC4wMSkgdW5pdF9wcmljZQpGUk9NIHJlZ2lvbiByCkpPSU4gc2FsZXNfcmVwcyBzCk9OIHMucmVnaW9uX2lkID0gci5pZApKT0lOIGFjY291bnRzIGEKT04gYS5zYWxlc19yZXBfaWQgPSBzLmlkCkpPSU4gb3JkZXJzIG8KT04gby5hY2NvdW50X2lkID0gYS5pZApXSEVSRSBvLnN0YW5kYXJkX3F0eSA+IDEwMDsKYGBgCgo1LiBQcm92aWRlIHRoZSBuYW1lIGZvciBlYWNoIHJlZ2lvbiBmb3IgZXZlcnkgb3JkZXIsIGFzIHdlbGwgYXMgdGhlIGFjY291bnQgbmFtZSBhbmQgdGhlIHVuaXQgcHJpY2UgdGhleSBwYWlkICh0b3RhbF9hbXRfdXNkL3RvdGFsKSBmb3IgdGhlIG9yZGVyLiBIb3dldmVyLCB5b3Ugc2hvdWxkIG9ubHkgcHJvdmlkZSB0aGUgcmVzdWx0cyBpZiB0aGUgc3RhbmRhcmQgb3JkZXIgcXVhbnRpdHkgZXhjZWVkcyAxMDAgYW5kIHRoZSBwb3N0ZXIgb3JkZXIgcXVhbnRpdHkgZXhjZWVkcyA1MC4gWW91ciBmaW5hbCB0YWJsZSBzaG91bGQgaGF2ZSAzIGNvbHVtbnM6IHJlZ2lvbiBuYW1lLCBhY2NvdW50IG5hbWUsIGFuZCB1bml0IHByaWNlLiBTb3J0IGZvciB0aGUgc21hbGxlc3QgdW5pdCBwcmljZSBmaXJzdC4gSW4gb3JkZXIgdG8gYXZvaWQgYSBkaXZpc2lvbiBieSB6ZXJvIGVycm9yLCBhZGRpbmcgLjAxIHRvIHRoZSBkZW5vbWluYXRvciBoZXJlIGlzIGhlbHBmdWwgKHRvdGFsX2FtdF91c2QvKHRvdGFsKzAuMDEpLiAKYGBge3NxbH0KU0VMRUNUIHIubmFtZSByZWdpb24sIGEubmFtZSBhY2NvdW50LCBvLnRvdGFsX2FtdF91c2QvKG8udG90YWwgKyAwLjAxKSB1bml0X3ByaWNlCkZST00gcmVnaW9uIHIKSk9JTiBzYWxlc19yZXBzIHMKT04gcy5yZWdpb25faWQgPSByLmlkCkpPSU4gYWNjb3VudHMgYQpPTiBhLnNhbGVzX3JlcF9pZCA9IHMuaWQKSk9JTiBvcmRlcnMgbwpPTiBvLmFjY291bnRfaWQgPSBhLmlkCldIRVJFIG8uc3RhbmRhcmRfcXR5ID4gMTAwIEFORCBvLnBvc3Rlcl9xdHkgPiA1MApPUkRFUiBCWSB1bml0X3ByaWNlOwpgYGAKCjYuIFByb3ZpZGUgdGhlIG5hbWUgZm9yIGVhY2ggcmVnaW9uIGZvciBldmVyeSBvcmRlciwgYXMgd2VsbCBhcyB0aGUgYWNjb3VudCBuYW1lIGFuZCB0aGUgdW5pdCBwcmljZSB0aGV5IHBhaWQgKHRvdGFsX2FtdF91c2QvdG90YWwpIGZvciB0aGUgb3JkZXIuIEhvd2V2ZXIsIHlvdSBzaG91bGQgb25seSBwcm92aWRlIHRoZSByZXN1bHRzIGlmIHRoZSBzdGFuZGFyZCBvcmRlciBxdWFudGl0eSBleGNlZWRzIDEwMCBhbmQgdGhlIHBvc3RlciBvcmRlciBxdWFudGl0eSBleGNlZWRzIDUwLiBZb3VyIGZpbmFsIHRhYmxlIHNob3VsZCBoYXZlIDMgY29sdW1uczogcmVnaW9uIG5hbWUsIGFjY291bnQgbmFtZSwgYW5kIHVuaXQgcHJpY2UuIFNvcnQgZm9yIHRoZSBsYXJnZXN0IHVuaXQgcHJpY2UgZmlyc3QuIEluIG9yZGVyIHRvIGF2b2lkIGEgZGl2aXNpb24gYnkgemVybyBlcnJvciwgYWRkaW5nIC4wMSB0byB0aGUgZGVub21pbmF0b3IgaGVyZSBpcyBoZWxwZnVsICh0b3RhbF9hbXRfdXNkLyh0b3RhbCswLjAxKS4gCmBgYHtzcWx9ClNFTEVDVCByLm5hbWUgcmVnaW9uLCBhLm5hbWUgYWNjb3VudCwgby50b3RhbF9hbXRfdXNkLyhvLnRvdGFsICsgMC4wMSkgdW5pdF9wcmljZQpGUk9NIHJlZ2lvbiByCkpPSU4gc2FsZXNfcmVwcyBzCk9OIHMucmVnaW9uX2lkID0gci5pZApKT0lOIGFjY291bnRzIGEKT04gYS5zYWxlc19yZXBfaWQgPSBzLmlkCkpPSU4gb3JkZXJzIG8KT04gby5hY2NvdW50X2lkID0gYS5pZApXSEVSRSBvLnN0YW5kYXJkX3F0eSA+IDEwMCBBTkQgby5wb3N0ZXJfcXR5ID4gNTAKT1JERVIgQlkgdW5pdF9wcmljZSBERVNDOwpgYGAKCjcuIFdoYXQgYXJlIHRoZSBkaWZmZXJlbnQgY2hhbm5lbHMgdXNlZCBieSBhY2NvdW50IGlkIDEwMDE/IFlvdXIgZmluYWwgdGFibGUgc2hvdWxkIGhhdmUgb25seSAyIGNvbHVtbnM6IGFjY291bnQgbmFtZSBhbmQgdGhlIGRpZmZlcmVudCBjaGFubmVscy4gWW91IGNhbiB0cnkgU0VMRUNUIERJU1RJTkNUIHRvIG5hcnJvdyBkb3duIHRoZSByZXN1bHRzIHRvIG9ubHkgdGhlIHVuaXF1ZSB2YWx1ZXMuCmBgYHtzcWx9ClNFTEVDVCBESVNUSU5DVCBhLm5hbWUsIHcuY2hhbm5lbApGUk9NIGFjY291bnRzIGEKUklHSFQgSk9JTiB3ZWJfZXZlbnRzIHcKT04gYS5pZCA9IHcuYWNjb3VudF9pZApXSEVSRSBhLmlkID0gJzEwMDEnOwpgYGAKCjguIEZpbmQgYWxsIHRoZSBvcmRlcnMgdGhhdCBvY2N1cnJlZCBpbiAyMDE1LiBZb3VyIGZpbmFsIHRhYmxlIHNob3VsZCBoYXZlIDQgY29sdW1uczogb2NjdXJyZWRfYXQsIGFjY291bnQgbmFtZSwgb3JkZXIgdG90YWwsIGFuZCBvcmRlciB0b3RhbF9hbXRfdXNkLiAKYGBge3NxbH0KU0VMRUNUIG8ub2NjdXJyZWRfYXQsIGEubmFtZSwgby50b3RhbCwgby50b3RhbF9hbXRfdXNkCkZST00gYWNjb3VudHMgYQpKT0lOIG9yZGVycyBvCk9OIG8uYWNjb3VudF9pZCA9IGEuaWQKV0hFUkUgby5vY2N1cnJlZF9hdCBCRVRXRUVOICcwMS0wMS0yMDE1JyBBTkQgJzAxLTAxLTIwMTYnCk9SREVSIEJZIG8ub2NjdXJyZWRfYXQgREVTQzsKYGBgCgoqKioKCiMgQUdHUkVHQVRJT05TCgojIyBOdWxsIHZhbHVlcwpXaGVuIGlkZW50aWZ5aW5nIE5VTExzIGluIGEgV0hFUkUgY2xhdXNlLCB3ZSB3cml0ZSBJUyBOVUxMIG9yIElTIE5PVCBOVUxMLiBXZSBkb24ndCB1c2UgPSwgYmVjYXVzZSBOVUxMIGlzbid0IGNvbnNpZGVyZWQgYSB2YWx1ZSBpbiBTUUwuCgojIyBDT1VOVApDb3VudCB0aGUgbnVtYmVyIG9mIHJvd3MgaW4gYSB0YWJsZQpgYGB7c3FsfQpTRUxFQ1QgQ09VTlQoKikKRlJPTSBhY2NvdW50czsKYGBgCkNPVU5UICpkb2VzIG5vdCogY29uc2lkZXIgcm93cyB0aGF0IGhhdmUgKk5VTEwqIHZhbHVlcy4gVGhlcmVmb3JlLCB0aGlzIGNhbiBiZSB1c2VmdWwgZm9yIHF1aWNrbHkgaWRlbnRpZnlpbmcgd2hpY2ggcm93cyBoYXZlIG1pc3NpbmcgZGF0YS4KCiMjIFNVTQpVbmxpa2UgQ09VTlQsIHlvdSBjYW4gb25seSB1c2UgU1VNIG9uIG51bWVyaWMgY29sdW1ucy4gSG93ZXZlciwgU1VNIHdpbGwgaWdub3JlIE5VTEwgdmFsdWVzIGFuZCB0cmVhdCB0aGVtIGFzIDAuCgojIyMgUXVlc3Rpb25zCjEuIEZpbmQgdGhlIHRvdGFsIGFtb3VudCBvZiBwb3N0ZXJfcXR5IHBhcGVyIG9yZGVyZWQgaW4gdGhlIG9yZGVycyB0YWJsZS4KYGBge3NxbH0KU0VMRUNUIFNVTShwb3N0ZXJfcXR5KSBBUyB0b3RhbF9wb3N0ZXJfc2FsZXMKRlJPTSBvcmRlcnM7CmBgYAoKMi4gRmluZCB0aGUgdG90YWwgYW1vdW50IG9mIHN0YW5kYXJkX3F0eSBwYXBlciBvcmRlcmVkIGluIHRoZSBvcmRlcnMgdGFibGUuCmBgYHtzcWx9ClNFTEVDVCBTVU0oc3RhbmRhcmRfcXR5KSBBUyB0b3RhbF9zdGFuZGFyZF9zYWxlcwpGUk9NIG9yZGVyczsKYGBgCgozLiBGaW5kIHRoZSB0b3RhbCBkb2xsYXIgYW1vdW50IG9mIHNhbGVzIHVzaW5nIHRoZSB0b3RhbF9hbXRfdXNkIGluIHRoZSBvcmRlcnMgdGFibGUuCmBgYHtzcWx9ClNFTEVDVCBTVU0odG90YWxfYW10X3VzZCkgQVMgdG90YWxfZG9sbGFyX3NhbGVzCkZST00gb3JkZXJzOwpgYGAKCjQuIEZpbmQgdGhlIHRvdGFsIGFtb3VudCBzcGVudCBvbiBzdGFuZGFyZF9hbXRfdXNkIGFuZCBnbG9zc19hbXRfdXNkIHBhcGVyIGZvciBlYWNoIG9yZGVyIGluIHRoZSBvcmRlcnMgdGFibGUuIFRoaXMgc2hvdWxkIGdpdmUgYSBkb2xsYXIgYW1vdW50IGZvciBlYWNoIG9yZGVyIGluIHRoZSB0YWJsZS4KKk5vdGljZSwgdGhpcyBzb2x1dGlvbiBkaWQgbm90IHVzZSBhbiBhZ2dyZWdhdGUuKgpgYGB7c3FsfQpTRUxFQ1Qgc3RhbmRhcmRfYW10X3VzZCArIGdsb3NzX2FtdF91c2QgQVMgdG90YWxfc3RhbmRhcmRfZ2xvc3MKRlJPTSBvcmRlcnM7CmBgYAoKNS4gRmluZCB0aGUgc3RhbmRhcmRfYW10X3VzZCBwZXIgdW5pdCBvZiBzdGFuZGFyZF9xdHkgcGFwZXIuIFlvdXIgc29sdXRpb24gc2hvdWxkIHVzZSBib3RoIGFuIGFnZ3JlZ2F0aW9uIGFuZCBhIG1hdGhlbWF0aWNhbCBvcGVyYXRvci4gCipOb3RpY2UsIHRoaXMgc29sdXRpb24gdXNlZCBib3RoIGFuIGFnZ3JlZ2F0ZSBhbmQgb3VyIG1hdGhlbWF0aWNhbCBvcGVyYXRvcnMqCmBgYHtzcWx9ClNFTEVDVCBTVU0oc3RhbmRhcmRfYW10X3VzZCkvU1VNKHN0YW5kYXJkX3F0eSkgQVMgc3RhbmRhcmRfcHJpY2VfcGVyX3VuaXQKRlJPTSBvcmRlcnM7CmBgYAoKIyMgTUlOIGFuZCBNQVgKTUlOIGFuZCBNQVggYXJlIGFnZ3JlZ2F0b3JzIHRoYXQgaWdub3JlIE5VTEwgdmFsdWVzLgpGdW5jdGlvbmFsbHksIE1JTiBhbmQgTUFYIGFyZSBzaW1pbGFyIHRvIENPVU5UIGluIHRoYXQgdGhleSBjYW4gYmUgdXNlZCBvbiBub24tbnVtZXJpY2FsIGNvbHVtbnMuIERlcGVuZGluZyBvbiB0aGUgY29sdW1uIHR5cGUsIE1JTiB3aWxsIHJldHVybiB0aGUgbG93ZXN0IG51bWJlciwgZWFybGllc3QgZGF0ZSwgb3Igbm9uLW51bWVyaWNhbCB2YWx1ZSBhcyBlYXJseSBpbiB0aGUgYWxwaGFiZXQgYXMgcG9zc2libGUuIEFzIHlvdSBtaWdodCBzdXNwZWN0LCBNQVggZG9lcyB0aGUgb3Bwb3NpdGXigJRpdCByZXR1cm5zIHRoZSBoaWdoZXN0IG51bWJlciwgdGhlIGxhdGVzdCBkYXRlLCBvciB0aGUgbm9uLW51bWVyaWNhbCB2YWx1ZSBjbG9zZXN0IGFscGhhYmV0aWNhbGx5IHRvIOKAnFou4oCdCgojIyMgUXVlc3Rpb25zCjEuIFdoZW4gd2FzIHRoZSBlYXJsaWVzdCBvcmRlciBldmVyIHBsYWNlZD8KYGBge3NxbH0KU0VMRUNUIE1JTihvY2N1cnJlZF9hdCkgCkZST00gb3JkZXJzOwpgYGAKVHJ5IHBlcmZvcm1pbmcgdGhlIHNhbWUgcXVlcnkgYXMgaW4gcXVlc3Rpb24gMSB3aXRob3V0IHVzaW5nIGFuIGFnZ3JlZ2F0aW9uIGZ1bmN0aW9uLiAKYGBge3NxbH0KU0VMRUNUIG9jY3VycmVkX2F0IApGUk9NIG9yZGVycyAKT1JERVIgQlkgb2NjdXJyZWRfYXQKTElNSVQgMTsKYGBgCgoyLiBXaGVuIGRpZCB0aGUgbW9zdCByZWNlbnQgKGxhdGVzdCkgd2ViX2V2ZW50IG9jY3VyPwpgYGB7c3FsfQpTRUxFQ1QgTUFYKG9jY3VycmVkX2F0KQpGUk9NIHdlYl9ldmVudHM7CmBgYApUcnkgdG8gcGVyZm9ybSB0aGUgcmVzdWx0IG9mIHRoZSBwcmV2aW91cyBxdWVyeSB3aXRob3V0IHVzaW5nIGFuIGFnZ3JlZ2F0aW9uIGZ1bmN0aW9uLgpgYGB7c3FsfQpTRUxFQ1Qgb2NjdXJyZWRfYXQKRlJPTSB3ZWJfZXZlbnRzCk9SREVSIEJZIG9jY3VycmVkX2F0IERFU0MKTElNSVQgMTsKYGBgCgojIyBBVkcKQVZHIHJldHVybnMgdGhlIG1lYW4gb2YgdGhlIGRhdGEgLSB0aGF0IGlzIHRoZSBzdW0gb2YgYWxsIG9mIHRoZSB2YWx1ZXMgaW4gdGhlIGNvbHVtbiBkaXZpZGVkIGJ5IHRoZSBudW1iZXIgb2YgdmFsdWVzIGluIGEgY29sdW1uLgpUaGlzIGFnZ3JlZ2F0ZSBmdW5jdGlvbiBpZ25vcmVzIHRoZSBOVUxMIHZhbHVlcyBpbiBib3RoIHRoZSBudW1lcmF0b3IgYW5kIHRoZSBkZW5vbWluYXRvci4KCiMjIyBRdWVzdGlvbnMKMS4gRmluZCB0aGUgbWVhbiAoQVZFUkFHRSkgYW1vdW50IHNwZW50IHBlciBvcmRlciBvbiBlYWNoIHBhcGVyIHR5cGUsIGFzIHdlbGwgYXMgdGhlIG1lYW4gYW1vdW50IG9mIGVhY2ggcGFwZXIgdHlwZSBwdXJjaGFzZWQgcGVyIG9yZGVyLiBZb3VyIGZpbmFsIGFuc3dlciBzaG91bGQgaGF2ZSA2IHZhbHVlcyAtIG9uZSBmb3IgZWFjaCBwYXBlciB0eXBlIGZvciB0aGUgYXZlcmFnZSBudW1iZXIgb2Ygc2FsZXMsIGFzIHdlbGwgYXMgdGhlIGF2ZXJhZ2UgYW1vdW50LgpgYGB7c3FsfQpTRUxFQ1QgQVZHKHN0YW5kYXJkX3F0eSkgbWVhbl9zdGFuZGFyZCwgQVZHKGdsb3NzX3F0eSkgbWVhbl9nbG9zcywgCiAgICAgICAgICAgQVZHKHBvc3Rlcl9xdHkpIG1lYW5fcG9zdGVyLCBBVkcoc3RhbmRhcmRfYW10X3VzZCkgbWVhbl9zdGFuZGFyZF91c2QsIAogICAgICAgICAgIEFWRyhnbG9zc19hbXRfdXNkKSBtZWFuX2dsb3NzX3VzZCwgQVZHKHBvc3Rlcl9hbXRfdXNkKSBtZWFuX3Bvc3Rlcl91c2QKRlJPTSBvcmRlcnM7CmBgYAoKMi4gV2hhdCBpcyB0aGUgTUVESUFOIHRvdGFsX3VzZCBzcGVudCBvbiBhbGwgb3JkZXJzPyBOb3RlLCB0aGlzIGlzIG1vcmUgYWR2YW5jZWQgdGhhbiB0aGUgdG9waWNzIHdlIGhhdmUgY292ZXJlZCB0aHVzIGZhciB0byBidWlsZCBhIGdlbmVyYWwgc29sdXRpb24sIGJ1dCB3ZSBjYW4gaGFyZCBjb2RlIGEgc29sdXRpb24gaW4gdGhlIGZvbGxvd2luZyB3YXkuCmBgYCB7c3FsfQpTRUxFQ1QgKgpGUk9NIChTRUxFQ1QgdG90YWxfYW10X3VzZAogICAgICBGUk9NIG9yZGVycwogICAgICBPUkRFUiBCWSB0b3RhbF9hbXRfdXNkCiAgICAgIExJTUlUIDM0NTcpIEFTIFRhYmxlMQpPUkRFUiBCWSB0b3RhbF9hbXRfdXNkIERFU0MKTElNSVQgMjsKYGBgClNpbmNlIHRoZXJlIGFyZSA2OTEyIG9yZGVycyAtIHdlIHdhbnQgdGhlIGF2ZXJhZ2Ugb2YgdGhlIDM0NTcgYW5kIDM0NTYgb3JkZXIgYW1vdW50cyB3aGVuIG9yZGVyZWQuIFRoaXMgaXMgdGhlIGF2ZXJhZ2Ugb2YgMjQ4My4xNiBhbmQgMjQ4Mi41NS4gVGhpcyBnaXZlcyB0aGUgbWVkaWFuIG9mIDI0ODIuODU1LiBUaGlzIG9idmlvdXNseSBpc24ndCBhbiBpZGVhbCB3YXkgdG8gY29tcHV0ZS4gSWYgd2Ugb2J0YWluIG5ldyBvcmRlcnMsIHdlIHdvdWxkIGhhdmUgdG8gY2hhbmdlIHRoZSBsaW1pdC4gU1FMIGRpZG4ndCBldmVuIGNhbGN1bGF0ZSB0aGUgbWVkaWFuIGZvciB1cy4gVGhlIGFib3ZlIHVzZWQgYSBTVUJRVUVSWSwgYnV0IHlvdSBjb3VsZCB1c2UgYW55IG1ldGhvZCB0byBmaW5kIHRoZSB0d28gbmVjZXNzYXJ5IHZhbHVlcywgYW5kIHRoZW4geW91IGp1c3QgbmVlZCB0aGUgYXZlcmFnZSBvZiB0aGVtLgoKIyMgR1JPVVAgQlkKR1JPVVAgQlkgY2FuIGJlIHVzZWQgdG8gYWdncmVnYXRlIGRhdGEgd2l0aGluIHN1YnNldHMgb2YgdGhlIGRhdGEuIEZvciBleGFtcGxlLCBncm91cGluZyBmb3IgZGlmZmVyZW50IGFjY291bnRzLCBkaWZmZXJlbnQgcmVnaW9ucywgb3IgZGlmZmVyZW50IHNhbGVzIHJlcHJlc2VudGF0aXZlcy4KQW55IGNvbHVtbiBpbiB0aGUgU0VMRUNUIHN0YXRlbWVudCB0aGF0IGlzIG5vdCB3aXRoaW4gYW4gYWdncmVnYXRvciBtdXN0IGJlIGluIHRoZSBHUk9VUCBCWSBjbGF1c2UuClRoZSBHUk9VUCBCWSBhbHdheXMgZ29lcyBiZXR3ZWVuIFdIRVJFIGFuZCBPUkRFUiBCWS4KT1JERVIgQlkgd29ya3MgbGlrZSBTT1JUIGluIHNwcmVhZHNoZWV0IHNvZnR3YXJlLgpTUUwgZXZhbHVhdGVzIHRoZSBhZ2dyZWdhdGlvbnMgYmVmb3JlIHRoZSBMSU1JVCBjbGF1c2UuIElmIHlvdSBkb27igJl0IGdyb3VwIGJ5IGFueSBjb2x1bW5zLCB5b3XigJlsbCBnZXQgYSAxLXJvdyByZXN1bHTigJRubyBwcm9ibGVtIHRoZXJlLiBJZiB5b3UgZ3JvdXAgYnkgYSBjb2x1bW4gd2l0aCBlbm91Z2ggdW5pcXVlIHZhbHVlcyB0aGF0IGl0IGV4Y2VlZHMgdGhlIExJTUlUIG51bWJlciwgdGhlIGFnZ3JlZ2F0ZXMgd2lsbCBiZSBjYWxjdWxhdGVkLCBhbmQgdGhlbiBzb21lIHJvd3Mgd2lsbCBzaW1wbHkgYmUgb21pdHRlZCBmcm9tIHRoZSByZXN1bHRzLgpUaGlzIGlzIGEgbmljZSB3YXkgdG8gZG8gdGhpbmdzIGJlY2F1c2UgeW91IGtub3cgeW914oCZcmUgZ29pbmcgdG8gZ2V0IHRoZSBjb3JyZWN0IGFnZ3JlZ2F0ZXMuIElmIFNRTCBjdXRzIHRoZSB0YWJsZSBkb3duIHRvIDEwMCByb3dzLCB0aGVuIHBlcmZvcm1lZCB0aGUgYWdncmVnYXRpb25zLCB5b3VyIHJlc3VsdHMgd291bGQgYmUgc3Vic3RhbnRpYWxseSBkaWZmZXJlbnQuCgpOb3RlOiBZb3UgY2FuIHJlZmVyZW5jZSB0aGUgY29sdW1ucyBpbiB5b3VyIHNlbGVjdCBzdGF0ZW1lbnQgaW4gR1JPVVAgQlkgYW5kIE9SREVSIEJZIGNsYXVzZXMgd2l0aCBudW1iZXJzIHRoYXQgZm9sbG93IHRoZSBvcmRlciB0aGV5IGFwcGVhciBpbiB0aGUgc2VsZWN0IHN0YXRlbWVudC4gRm9yIGV4YW1wbGUKYGBge3NxbH0KU0VMRUNUIHN0YW5kYXJkX3F0eSwgQ09VTlQoKikKCkZST00gb3JkZXJzCgpHUk9VUCBCWSAxIC8qKHRoaXMgMSByZWZlcnMgdG8gc3RhbmRhcmRfcXR5IHNpbmNlIGl0IGlzIHRoZSBmaXJzdCBvZiB0aGUgY29sdW1ucyBpbmNsdWRlZCBpbiB0aGUgc2VsZWN0IHN0YXRlbWVudCkqLwoKT1JERVIgQlkgMSAvKih0aGlzIDEgcmVmZXJzIHRvIHN0YW5kYXJkX3F0eSBzaW5jZSBpdCBpcyB0aGUgZmlyc3Qgb2YgdGhlIGNvbHVtbnMgaW5jbHVkZWQgaW4gdGhlIHNlbGVjdCBzdGF0ZW1lbnQpKi8KYGBgCgojIyMgUXVlc3Rpb25zCjEuIFdoaWNoIGFjY291bnQgKGJ5IG5hbWUpIHBsYWNlZCB0aGUgZWFybGllc3Qgb3JkZXI/IFlvdXIgc29sdXRpb24gc2hvdWxkIGhhdmUgdGhlIGFjY291bnQgbmFtZSBhbmQgdGhlIGRhdGUgb2YgdGhlIG9yZGVyLgpgYGB7c3FsfQpTRUxFQ1QgYS5uYW1lLCBvLm9jY3VycmVkX2F0CkZST00gYWNjb3VudHMgYQpKT0lOIG9yZGVycyBvCk9OIGEuaWQgPSBvLmFjY291bnRfaWQKT1JERVIgQlkgb2NjdXJyZWRfYXQKTElNSVQgMTsKYGBgCgoyLiBGaW5kIHRoZSB0b3RhbCBzYWxlcyBpbiB1c2QgZm9yIGVhY2ggYWNjb3VudC4gWW91IHNob3VsZCBpbmNsdWRlIHR3byBjb2x1bW5zIC0gdGhlIHRvdGFsIHNhbGVzIGZvciBlYWNoIGNvbXBhbnkncyBvcmRlcnMgaW4gdXNkIGFuZCB0aGUgY29tcGFueSBuYW1lLgpgYGB7c3FsfQpTRUxFQ1QgYS5uYW1lLCBTVU0odG90YWxfYW10X3VzZCkgdG90YWxfc2FsZXMKRlJPTSBvcmRlcnMgbwpKT0lOIGFjY291bnRzIGEKT04gYS5pZCA9IG8uYWNjb3VudF9pZApHUk9VUCBCWSBhLm5hbWU7CmBgYAoKMy4gVmlhIHdoYXQgY2hhbm5lbCBkaWQgdGhlIG1vc3QgcmVjZW50IChsYXRlc3QpIHdlYl9ldmVudCBvY2N1ciwgd2hpY2ggYWNjb3VudCB3YXMgYXNzb2NpYXRlZCB3aXRoIHRoaXMgd2ViX2V2ZW50PyBZb3VyIHF1ZXJ5IHNob3VsZCByZXR1cm4gb25seSB0aHJlZSB2YWx1ZXMgLSB0aGUgZGF0ZSwgY2hhbm5lbCwgYW5kIGFjY291bnQgbmFtZS4KYGBge3NxbH0KU0VMRUNUIHcub2NjdXJyZWRfYXQsIHcuY2hhbm5lbCwgYS5uYW1lCkZST00gd2ViX2V2ZW50cyB3CkpPSU4gYWNjb3VudHMgYQpPTiB3LmFjY291bnRfaWQgPSBhLmlkIApPUkRFUiBCWSB3Lm9jY3VycmVkX2F0IERFU0MKTElNSVQgMTsKYGBgCgo0LiBGaW5kIHRoZSB0b3RhbCBudW1iZXIgb2YgdGltZXMgZWFjaCB0eXBlIG9mIGNoYW5uZWwgZnJvbSB0aGUgd2ViX2V2ZW50cyB3YXMgdXNlZC4gWW91ciBmaW5hbCB0YWJsZSBzaG91bGQgaGF2ZSB0d28gY29sdW1ucyAtIHRoZSBjaGFubmVsIGFuZCB0aGUgbnVtYmVyIG9mIHRpbWVzIHRoZSBjaGFubmVsIHdhcyB1c2VkLgpgYGB7c3FsfQpTRUxFQ1Qgdy5jaGFubmVsLCBDT1VOVCgqKQpGUk9NIHdlYl9ldmVudHMgdwpHUk9VUCBCWSB3LmNoYW5uZWwKYGBgCgo1LiBXaG8gd2FzIHRoZSBwcmltYXJ5IGNvbnRhY3QgYXNzb2NpYXRlZCB3aXRoIHRoZSBlYXJsaWVzdCB3ZWJfZXZlbnQ/CmBgYHtzcWx9ClNFTEVDVCBhLnByaW1hcnlfcG9jCkZST00gd2ViX2V2ZW50cyB3CkpPSU4gYWNjb3VudHMgYQpPTiBhLmlkID0gdy5hY2NvdW50X2lkCk9SREVSIEJZIHcub2NjdXJyZWRfYXQKTElNSVQgMTsKYGBgCgo2LiBXaGF0IHdhcyB0aGUgc21hbGxlc3Qgb3JkZXIgcGxhY2VkIGJ5IGVhY2ggYWNjb3VudCBpbiB0ZXJtcyBvZiB0b3RhbCB1c2QuIFByb3ZpZGUgb25seSB0d28gY29sdW1ucyAtIHRoZSBhY2NvdW50IG5hbWUgYW5kIHRoZSB0b3RhbCB1c2QuIE9yZGVyIGZyb20gc21hbGxlc3QgZG9sbGFyIGFtb3VudHMgdG8gbGFyZ2VzdC4KYGBge3NxbH0gClNFTEVDVCBhLm5hbWUsIE1JTih0b3RhbF9hbXRfdXNkKSBzbWFsbGVzdF9vcmRlcgpGUk9NIGFjY291bnRzIGEKSk9JTiBvcmRlcnMgbwpPTiBhLmlkID0gby5hY2NvdW50X2lkCkdST1VQIEJZIGEubmFtZQpPUkRFUiBCWSBzbWFsbGVzdF9vcmRlcjsKYGBgClNvcnQgb2Ygc3RyYW5nZSB3ZSBoYXZlIGEgYnVuY2ggb2Ygb3JkZXJzIHdpdGggbm8gZG9sbGFycy4gV2UgbWlnaHQgd2FudCB0byBsb29rIGludG8gdGhvc2UuIAoKNy4gRmluZCB0aGUgbnVtYmVyIG9mIHNhbGVzIHJlcHMgaW4gZWFjaCByZWdpb24uIFlvdXIgZmluYWwgdGFibGUgc2hvdWxkIGhhdmUgdHdvIGNvbHVtbnMgLSB0aGUgcmVnaW9uIGFuZCB0aGUgbnVtYmVyIG9mIHNhbGVzX3JlcHMuIE9yZGVyIGZyb20gZmV3ZXN0IHJlcHMgdG8gbW9zdCByZXBzLgpgYGB7c3FsfQpTRUxFQ1Qgci5uYW1lLCBDT1VOVCgqKSBudW1fcmVwcwpGUk9NIHJlZ2lvbiByCkpPSU4gc2FsZXNfcmVwcyBzCk9OIHIuaWQgPSBzLnJlZ2lvbl9pZApHUk9VUCBCWSByLm5hbWUKT1JERVIgQlkgbnVtX3JlcHM7CmBgYAoKWW91IGNhbiBHUk9VUCBCWSBtdWx0aXBsZSBjb2x1bW5zIGF0IG9uY2UuIFRoaXMgaXMgb2Z0ZW4gdXNlZnVsIHRvIGFnZ3JlZ2F0ZSBhY3Jvc3MgYSBudW1iZXIgb2YgZGlmZmVyZW50IHNlZ21lbnRzLiAKVGhlIG9yZGVyIG9mIGNvbHVtbnMgbGlzdGVkIGluIHRoZSAqT1JERVIgQlkqIGNsYXVzZSBkb2VzIG1ha2UgYSBkaWZmZXJlbmNlLiBZb3UgYXJlIG9yZGVyaW5nIHRoZSBjb2x1bW5zIGZyb20gbGVmdCB0byByaWdodC4KVGhlIG9yZGVyIG9mIGNvbHVtbiBuYW1lcyBpbiB5b3VyICpHUk9VUCBCWSogY2xhdXNlIGRvZXNu4oCZdCBtYXR0ZXLigJR0aGUgcmVzdWx0cyB3aWxsIGJlIHRoZSBzYW1lIHJlZ2FyZGxlc3MuCgojIyMgUXVlc3Rpb25zCjEuIEZvciBlYWNoIGFjY291bnQsIGRldGVybWluZSB0aGUgYXZlcmFnZSBhbW91bnQgb2YgZWFjaCB0eXBlIG9mIHBhcGVyIHRoZXkgcHVyY2hhc2VkIGFjcm9zcyB0aGVpciBvcmRlcnMuIFlvdXIgcmVzdWx0IHNob3VsZCBoYXZlIGZvdXIgY29sdW1ucyAtIG9uZSBmb3IgdGhlIGFjY291bnQgbmFtZSBhbmQgb25lIGZvciB0aGUgYXZlcmFnZSBzcGVudCBvbiBlYWNoIG9mIHRoZSBwYXBlciB0eXBlcy4KYGBge3NxbH0KU0VMRUNUIGEubmFtZSwgQVZHKG8uc3RhbmRhcmRfcXR5KSBhdmdfc3RhbmQsIEFWRyhvLmdsb3NzX3F0eSkgYXZnX2dsb3NzLCBBVkcoby5wb3N0ZXJfcXR5KSBhdmdfcG9zdApGUk9NIGFjY291bnRzIGEKSk9JTiBvcmRlcnMgbwpPTiBhLmlkID0gby5hY2NvdW50X2lkCkdST1VQIEJZIGEubmFtZTsKYGBgCgoyLiBGb3IgZWFjaCBhY2NvdW50LCBkZXRlcm1pbmUgdGhlIGF2ZXJhZ2UgYW1vdW50IHNwZW50IHBlciBvcmRlciBvbiBlYWNoIHBhcGVyIHR5cGUuIFlvdXIgcmVzdWx0IHNob3VsZCBoYXZlIGZvdXIgY29sdW1ucyAtIG9uZSBmb3IgdGhlIGFjY291bnQgbmFtZSBhbmQgb25lIGZvciB0aGUgYXZlcmFnZSBhbW91bnQgc3BlbnQgb24gZWFjaCBwYXBlciB0eXBlLgpgYGB7c3FsfQpTRUxFQ1QgYS5uYW1lLCBBVkcoby5zdGFuZGFyZF9hbXRfdXNkKSBhdmdfc3RhbmQsIEFWRyhvLmdsb3NzX2FtdF91c2QpIGF2Z19nbG9zcywgQVZHKG8ucG9zdGVyX2FtdF91c2QpIGF2Z19wb3N0CkZST00gYWNjb3VudHMgYQpKT0lOIG9yZGVycyBvCk9OIGEuaWQgPSBvLmFjY291bnRfaWQKR1JPVVAgQlkgYS5uYW1lOwpgYGAKCjMuIERldGVybWluZSB0aGUgbnVtYmVyIG9mIHRpbWVzIGEgcGFydGljdWxhciBjaGFubmVsIHdhcyB1c2VkIGluIHRoZSB3ZWJfZXZlbnRzIHRhYmxlIGZvciBlYWNoIHNhbGVzIHJlcC4gWW91ciBmaW5hbCB0YWJsZSBzaG91bGQgaGF2ZSB0aHJlZSBjb2x1bW5zIC0gdGhlIG5hbWUgb2YgdGhlIHNhbGVzIHJlcCwgdGhlIGNoYW5uZWwsIGFuZCB0aGUgbnVtYmVyIG9mIG9jY3VycmVuY2VzLiBPcmRlciB5b3VyIHRhYmxlIHdpdGggdGhlIGhpZ2hlc3QgbnVtYmVyIG9mIG9jY3VycmVuY2VzIGZpcnN0LgpgYGB7c3FsfQpTRUxFQ1Qgcy5uYW1lLCB3LmNoYW5uZWwsIENPVU5UKCopIG51bV9ldmVudHMKRlJPTSBhY2NvdW50cyBhCkpPSU4gd2ViX2V2ZW50cyB3Ck9OIGEuaWQgPSB3LmFjY291bnRfaWQKSk9JTiBzYWxlc19yZXBzIHMKT04gcy5pZCA9IGEuc2FsZXNfcmVwX2lkCkdST1VQIEJZIHMubmFtZSwgdy5jaGFubmVsCk9SREVSIEJZIG51bV9ldmVudHMgREVTQzsKYGBgCgo0LiBEZXRlcm1pbmUgdGhlIG51bWJlciBvZiB0aW1lcyBhIHBhcnRpY3VsYXIgY2hhbm5lbCB3YXMgdXNlZCBpbiB0aGUgd2ViX2V2ZW50cyB0YWJsZSBmb3IgZWFjaCByZWdpb24uIFlvdXIgZmluYWwgdGFibGUgc2hvdWxkIGhhdmUgdGhyZWUgY29sdW1ucyAtIHRoZSByZWdpb24gbmFtZSwgdGhlIGNoYW5uZWwsIGFuZCB0aGUgbnVtYmVyIG9mIG9jY3VycmVuY2VzLiBPcmRlciB5b3VyIHRhYmxlIHdpdGggdGhlIGhpZ2hlc3QgbnVtYmVyIG9mIG9jY3VycmVuY2VzIGZpcnN0LgpgYGB7c3FsfQpTRUxFQ1Qgci5uYW1lLCB3LmNoYW5uZWwsIENPVU5UKCopIG51bV9ldmVudHMKRlJPTSBhY2NvdW50cyBhCkpPSU4gd2ViX2V2ZW50cyB3Ck9OIGEuaWQgPSB3LmFjY291bnRfaWQKSk9JTiBzYWxlc19yZXBzIHMKT04gcy5pZCA9IGEuc2FsZXNfcmVwX2lkCkpPSU4gcmVnaW9uIHIKT04gci5pZCA9IHMucmVnaW9uX2lkCkdST1VQIEJZIHIubmFtZSwgdy5jaGFubmVsCk9SREVSIEJZIG51bV9ldmVudHMgREVTQzsKYGBgCgojIyBESVNUSU5DVCAKRElTVElOQ1QgaXMgYWx3YXlzIHVzZWQgaW4gU0VMRUNUIHN0YXRlbWVudHMsIGFuZCBpdCBwcm92aWRlcyB0aGUgdW5pcXVlIHJvd3MgZm9yIGFsbCBjb2x1bW5zIHdyaXR0ZW4gaW4gdGhlIFNFTEVDVCBzdGF0ZW1lbnQuIFRoZXJlZm9yZSwgeW91IG9ubHkgdXNlIERJU1RJTkNUIG9uY2UgaW4gYW55IHBhcnRpY3VsYXIgU0VMRUNUIHN0YXRlbWVudC4KYGBge3NxbH0KU0VMRUNUIERJU1RJTkNUIGNvbHVtbjEsIGNvbHVtbjIsIGNvbHVtbjMKRlJPTSB0YWJsZTE7CmBgYAp3aGljaCB3b3VsZCByZXR1cm4gdGhlIHVuaXF1ZSAob3IgRElTVElOQ1QpIHJvd3MgYWNyb3NzIGFsbCB0aHJlZSBjb2x1bW5zLgpOb3RlOiB1c2luZyBESVNUSU5DVCwgcGFydGljdWxhcmx5IGluIGFnZ3JlZ2F0aW9ucywgY2FuIHNsb3cgeW91ciBxdWVyaWVzIGRvd24gcXVpdGUgYSBiaXQuCgojIyMgUXVlc3Rpb25zCjEuIFVzZSBESVNUSU5DVCB0byB0ZXN0IGlmIHRoZXJlIGFyZSBhbnkgYWNjb3VudHMgYXNzb2NpYXRlZCB3aXRoIG1vcmUgdGhhbiBvbmUgcmVnaW9uLgoKVGhlIGJlbG93IHR3byBxdWVyaWVzIGhhdmUgdGhlIHNhbWUgbnVtYmVyIG9mIHJlc3VsdGluZyByb3dzICgzNTEpLCBzbyB3ZSBrbm93IHRoYXQgZXZlcnkgYWNjb3VudCBpcyBhc3NvY2lhdGVkIHdpdGggb25seSBvbmUgcmVnaW9uLiBJZiBlYWNoIGFjY291bnQgd2FzIGFzc29jaWF0ZWQgd2l0aCBtb3JlIHRoYW4gb25lIHJlZ2lvbiwgdGhlIGZpcnN0IHF1ZXJ5IHNob3VsZCBoYXZlIHJldHVybmVkIG1vcmUgcm93cyB0aGFuIHRoZSBzZWNvbmQgcXVlcnkuCmBgYHtzcWx9ClNFTEVDVCBhLmlkIGFzICJhY2NvdW50IGlkIiwgci5pZCBhcyAicmVnaW9uIGlkIiwgCmEubmFtZSBhcyAiYWNjb3VudCBuYW1lIiwgci5uYW1lIGFzICJyZWdpb24gbmFtZSIKRlJPTSBhY2NvdW50cyBhCkpPSU4gc2FsZXNfcmVwcyBzCk9OIHMuaWQgPSBhLnNhbGVzX3JlcF9pZApKT0lOIHJlZ2lvbiByCk9OIHIuaWQgPSBzLnJlZ2lvbl9pZDsKYGBgCmFuZApgYGB7c3FsfQpTRUxFQ1QgRElTVElOQ1QgaWQsIG5hbWUKRlJPTSBhY2NvdW50czsKYGBgCjIuIEhhdmUgYW55IHNhbGVzIHJlcHMgd29ya2VkIG9uIG1vcmUgdGhhbiBvbmUgYWNjb3VudD8KCkFjdHVhbGx5IGFsbCBvZiB0aGUgc2FsZXMgcmVwcyBoYXZlIHdvcmtlZCBvbiBtb3JlIHRoYW4gb25lIGFjY291bnQuIFRoZSBmZXdlc3QgbnVtYmVyIG9mIGFjY291bnRzIGFueSBzYWxlcyByZXAgd29ya3Mgb24gaXMgMy4gVGhlcmUgYXJlIDUwIHNhbGVzIHJlcHMsIGFuZCB0aGV5IGFsbCBoYXZlIG1vcmUgdGhhbiBvbmUgYWNjb3VudC4gVXNpbmcgRElTVElOQ1QgaW4gdGhlIHNlY29uZCBxdWVyeSBhc3N1cmVzIHRoYXQgYWxsIG9mIHRoZSBzYWxlcyByZXBzIGFyZSBhY2NvdW50ZWQgZm9yIGluIHRoZSBmaXJzdCBxdWVyeS4KYGBge3NxbH0KU0VMRUNUIHMuaWQsIHMubmFtZSwgQ09VTlQoKikgbnVtX2FjY291bnRzCkZST00gYWNjb3VudHMgYQpKT0lOIHNhbGVzX3JlcHMgcwpPTiBzLmlkID0gYS5zYWxlc19yZXBfaWQKR1JPVVAgQlkgcy5pZCwgcy5uYW1lCk9SREVSIEJZIG51bV9hY2NvdW50czsKYGBgCgpgYGB7c3FsfQpTRUxFQ1QgRElTVElOQ1QgaWQsIG5hbWUKRlJPTSBzYWxlc19yZXBzOwpgYGAKCiMjIEhBVklORwpIQVZJTkcgaXMgdGhlIOKAnGNsZWFu4oCdIHdheSB0byBmaWx0ZXIgYSBxdWVyeSB0aGF0IGhhcyBiZWVuIGFnZ3JlZ2F0ZWQsIGJ1dCB0aGlzIGlzIGFsc28gY29tbW9ubHkgZG9uZSB1c2luZyBhIHN1YnF1ZXJ5LiBFc3NlbnRpYWxseSwgYW55IHRpbWUgeW91IHdhbnQgdG8gcGVyZm9ybSBhIFdIRVJFIG9uIGFuIGVsZW1lbnQgb2YgeW91ciBxdWVyeSB0aGF0IHdhcyBjcmVhdGVkIGJ5IGFuIGFnZ3JlZ2F0ZSwgeW91IG5lZWQgdG8gdXNlIEhBVklORyBpbnN0ZWFkLgpIQVZJTkcgYXBwZWFycyBhZnRlciB0aGUgR1JPVVAgQlkgY2xhdXNlIGJ1dCBiZWZvcmUgdGhlIE9SREVSIEJZIGNsYXVzZS4KCiMjIyBRdWVzdGlvbnMKMS4gSG93IG1hbnkgb2YgdGhlIHNhbGVzIHJlcHMgaGF2ZSBtb3JlIHRoYW4gNSBhY2NvdW50cyB0aGF0IHRoZXkgbWFuYWdlPwpgYGB7c3FsfQpTRUxFQ1Qgcy5pZCwgcy5uYW1lLCBDT1VOVCgqKSBudW1fYWNjb3VudHMKRlJPTSBhY2NvdW50cyBhCkpPSU4gc2FsZXNfcmVwcyBzCk9OIHMuaWQgPSBhLnNhbGVzX3JlcF9pZApHUk9VUCBCWSBzLmlkLCBzLm5hbWUKSEFWSU5HIENPVU5UKCopID4gNQpPUkRFUiBCWSBudW1fYWNjb3VudHM7CmBgYAoKMi4gSG93IG1hbnkgYWNjb3VudHMgaGF2ZSBtb3JlIHRoYW4gMjAgb3JkZXJzPwpgYGB7c3FsfQpTRUxFQ1QgYS5pZCwgYS5uYW1lLCBDT1VOVCgqKSBudW1fb3JkZXJzCkZST00gYWNjb3VudHMgYQpKT0lOIG9yZGVycyBvCk9OIGEuaWQgPSBvLmFjY291bnRfaWQKR1JPVVAgQlkgYS5pZCwgYS5uYW1lCkhBVklORyBDT1VOVCgqKSA+IDIwCk9SREVSIEJZIG51bV9vcmRlcnM7CmBgYAoKMy4gV2hpY2ggYWNjb3VudCBoYXMgdGhlIG1vc3Qgb3JkZXJzPwpgYGB7c3FsfQpTRUxFQ1QgYS5pZCwgYS5uYW1lLCBDT1VOVCgqKSBudW1fb3JkZXJzCkZST00gYWNjb3VudHMgYQpKT0lOIG9yZGVycyBvCk9OIGEuaWQgPSBvLmFjY291bnRfaWQKR1JPVVAgQlkgYS5pZCwgYS5uYW1lCk9SREVSIEJZIG51bV9vcmRlcnMgREVTQwpMSU1JVCAxOwpgYGAKCjQuIEhvdyBtYW55IGFjY291bnRzIHNwZW50IG1vcmUgdGhhbiAzMCwwMDAgdXNkIHRvdGFsIGFjcm9zcyBhbGwgb3JkZXJzPwpgYGB7c3FsfQpTRUxFQ1QgYS5pZCwgYS5uYW1lLCBTVU0oby50b3RhbF9hbXRfdXNkKSB0b3RhbF9zcGVudApGUk9NIGFjY291bnRzIGEKSk9JTiBvcmRlcnMgbwpPTiBhLmlkID0gby5hY2NvdW50X2lkCkdST1VQIEJZIGEuaWQsIGEubmFtZQpIQVZJTkcgU1VNKG8udG90YWxfYW10X3VzZCkgPiAzMDAwMApPUkRFUiBCWSB0b3RhbF9zcGVudDsKYGBgCgo1LiBIb3cgbWFueSBhY2NvdW50cyBzcGVudCBsZXNzIHRoYW4gMSwwMDAgdXNkIHRvdGFsIGFjcm9zcyBhbGwgb3JkZXJzPwpgYGB7c3FsfQpTRUxFQ1QgYS5pZCwgYS5uYW1lLCBTVU0oby50b3RhbF9hbXRfdXNkKSB0b3RhbF9zcGVudApGUk9NIGFjY291bnRzIGEKSk9JTiBvcmRlcnMgbwpPTiBhLmlkID0gby5hY2NvdW50X2lkCkdST1VQIEJZIGEuaWQsIGEubmFtZQpIQVZJTkcgU1VNKG8udG90YWxfYW10X3VzZCkgPCAxMDAwCk9SREVSIEJZIHRvdGFsX3NwZW50OwpgYGAKCjYuIFdoaWNoIGFjY291bnQgaGFzIHNwZW50IHRoZSBtb3N0IHdpdGggdXM/CmBgYHtzcWx9ClNFTEVDVCBhLmlkLCBhLm5hbWUsIFNVTShvLnRvdGFsX2FtdF91c2QpIHRvdGFsX3NwZW50CkZST00gYWNjb3VudHMgYQpKT0lOIG9yZGVycyBvCk9OIGEuaWQgPSBvLmFjY291bnRfaWQKR1JPVVAgQlkgYS5pZCwgYS5uYW1lCk9SREVSIEJZIHRvdGFsX3NwZW50IERFU0MKTElNSVQgMTsKYGBgCgo3LiBXaGljaCBhY2NvdW50IGhhcyBzcGVudCB0aGUgbGVhc3Qgd2l0aCB1cz8KYGBge3NxbH0KU0VMRUNUIGEuaWQsIGEubmFtZSwgU1VNKG8udG90YWxfYW10X3VzZCkgdG90YWxfc3BlbnQKRlJPTSBhY2NvdW50cyBhCkpPSU4gb3JkZXJzIG8KT04gYS5pZCA9IG8uYWNjb3VudF9pZApHUk9VUCBCWSBhLmlkLCBhLm5hbWUKT1JERVIgQlkgdG90YWxfc3BlbnQKTElNSVQgMTsKYGBgCgo4LiBXaGljaCBhY2NvdW50cyB1c2VkIGZhY2Vib29rIGFzIGEgY2hhbm5lbCB0byBjb250YWN0IGN1c3RvbWVycyBtb3JlIHRoYW4gNiB0aW1lcz8KYGBge3NxbH0KU0VMRUNUIGEuaWQsIGEubmFtZSwgdy5jaGFubmVsLCBDT1VOVCgqKSB1c2Vfb2ZfY2hhbm5lbApGUk9NIGFjY291bnRzIGEKSk9JTiB3ZWJfZXZlbnRzIHcKT04gYS5pZCA9IHcuYWNjb3VudF9pZApHUk9VUCBCWSBhLmlkLCBhLm5hbWUsIHcuY2hhbm5lbApIQVZJTkcgQ09VTlQoKikgPiA2IEFORCB3LmNoYW5uZWwgPSAnZmFjZWJvb2snCk9SREVSIEJZIHVzZV9vZl9jaGFubmVsOwpgYGAKCjkuIFdoaWNoIGFjY291bnQgdXNlZCBmYWNlYm9vayBtb3N0IGFzIGEgY2hhbm5lbD8gCmBgYHtzcWx9ClNFTEVDVCBhLmlkLCBhLm5hbWUsIHcuY2hhbm5lbCwgQ09VTlQoKikgdXNlX29mX2NoYW5uZWwKRlJPTSBhY2NvdW50cyBhCkpPSU4gd2ViX2V2ZW50cyB3Ck9OIGEuaWQgPSB3LmFjY291bnRfaWQKV0hFUkUgdy5jaGFubmVsID0gJ2ZhY2Vib29rJwpHUk9VUCBCWSBhLmlkLCBhLm5hbWUsIHcuY2hhbm5lbApPUkRFUiBCWSB1c2Vfb2ZfY2hhbm5lbCBERVNDCkxJTUlUIDE7CmBgYAoKMTAuIFdoaWNoIGNoYW5uZWwgd2FzIG1vc3QgZnJlcXVlbnRseSB1c2VkIGJ5IG1vc3QgYWNjb3VudHM/CmBgYHtzcWx9ClNFTEVDVCBhLmlkLCBhLm5hbWUsIHcuY2hhbm5lbCwgQ09VTlQoKikgdXNlX29mX2NoYW5uZWwKRlJPTSBhY2NvdW50cyBhCkpPSU4gd2ViX2V2ZW50cyB3Ck9OIGEuaWQgPSB3LmFjY291bnRfaWQKR1JPVVAgQlkgYS5pZCwgYS5uYW1lLCB3LmNoYW5uZWwKT1JERVIgQlkgdXNlX29mX2NoYW5uZWwgREVTQwpMSU1JVCAxMDsKYGBgCgojIyBEQVRFIEZ1bmN0aW9ucwoqR1JPVVAqaW5nICpCWSogYSBkYXRlIGNvbHVtbiBpcyBub3QgdXN1YWxseSB2ZXJ5IHVzZWZ1bCBpbiBTUUwsIGFzIHRoZXNlIGNvbHVtbnMgdGVuZCB0byBoYXZlIHRyYW5zYWN0aW9uIGRhdGEgZG93biB0byBhIHNlY29uZC4gS2VlcGluZyBkYXRlIGluZm9ybWF0aW9uIGF0IHN1Y2ggYSBncmFudWxhciBkYXRhIGlzIGJvdGggYSBibGVzc2luZyBhbmQgYSBjdXJzZSwgYXMgaXQgZ2l2ZXMgcmVhbGx5IHByZWNpc2UgaW5mb3JtYXRpb24gKGEgYmxlc3NpbmcpLCBidXQgaXQgbWFrZXMgZ3JvdXBpbmcgaW5mb3JtYXRpb24gdG9nZXRoZXIgZGlyZWN0bHkgZGlmZmljdWx0IChhIGN1cnNlKS4KRGF0ZXMgYXJlIHN0b3JlZCBpbiB5ZWFyLCBtb250aCwgZGF5LCBob3VyLCBtaW51dGUsIHNlY29uZCwgd2hpY2ggaGVscHMgdXMgaW4gdHJ1bmNhdGluZy4KVGhlcmUgYXJlIGEgbnVtYmVyIG9mIGJ1aWx0IGluIFNRTCBmdW5jdGlvbnMgdGhhdCBhcmUgYWltZWQgYXQgaGVscGluZyB1cyBpbXByb3ZlIG91ciBleHBlcmllbmNlIGluIHdvcmtpbmcgd2l0aCBkYXRlczoKSW4gU1FMIGRhdGVzIGFyZSBzdG9yZWQgaW4gdGhlIGZvcm1hdCBZWVlZLU1NLUREIHdpdGggdGhlIHRpbWUgaW4gaG91cnMsIG1pbnV0ZXMgYW5kIHNlY29uZHMgYXQgdGhlIGVuZC4gCipEQVRFX1RSVU5DKiBhbGxvd3MgeW91IHRvIHRydW5jYXRlIHlvdXIgZGF0ZSB0byBhIHBhcnRpY3VsYXIgcGFydCBvZiB5b3VyIGRhdGUtdGltZSBjb2x1bW4uIENvbW1vbiB0cnVuY3Rpb25zIGFyZSBkYXksIG1vbnRoLCBhbmQgeWVhci4KKkRBVEVfUEFSVCogY2FuIGJlIHVzZWZ1bCBmb3IgcHVsbGluZyBhIHNwZWNpZmljIHBvcnRpb24gb2YgYSBkYXRlLCBidXQgbm90aWNlIHB1bGxpbmcgbW9udGggb3IgZGF5IG9mIHRoZSB3ZWVrIChkb3cpIG1lYW5zIHRoYXQgeW91IGFyZSBubyBsb25nZXIga2VlcGluZyB0aGUgeWVhcnMgaW4gb3JkZXIuIFJhdGhlciB5b3UgYXJlIGdyb3VwaW5nIGZvciBjZXJ0YWluIGNvbXBvbmVudHMgcmVnYXJkbGVzcyBvZiB3aGljaCB5ZWFyIHRoZXkgYmVsb25nZWQgaW4uCipOb3RlOiogRE9XIHJldHVybnMgYSB2YWx1ZSBmcm9tIDAtNiB3aGVyZSAwIGlzIFN1bmRheSBhbmQgNiBpcyBTYXR1cmRheS4gCgojIyMgUXVlc3Rpb25zCjEuIEZpbmQgdGhlIHNhbGVzIGluIHRlcm1zIG9mIHRvdGFsIGRvbGxhcnMgZm9yIGFsbCBvcmRlcnMgaW4gZWFjaCAqeWVhciosIG9yZGVyZWQgZnJvbSBncmVhdGVzdCB0byBsZWFzdC4gRG8geW91IG5vdGljZSBhbnkgdHJlbmRzIGluIHRoZSB5ZWFybHkgc2FsZXMgdG90YWxzPwpgYGB7c3FsfQpTRUxFQ1QgREFURV9QQVJUKCd5ZWFyJywgb2NjdXJyZWRfYXQpIG9yZF95ZWFyLCAgU1VNKHRvdGFsX2FtdF91c2QpIHRvdGFsX3NwZW50CkZST00gb3JkZXJzCkdST1VQIEJZIDEKT1JERVIgQlkgMiBERVNDOwpgYGAKV2hlbiB3ZSBsb29rIGF0IHRoZSB5ZWFybHkgdG90YWxzLCB5b3UgbWlnaHQgbm90aWNlIHRoYXQgMjAxMyBhbmQgMjAxNyBoYXZlIG11Y2ggc21hbGxlciB0b3RhbHMgdGhhbiBhbGwgb3RoZXIgeWVhcnMuIElmIHdlIGxvb2sgZnVydGhlciBhdCB0aGUgbW9udGhseSBkYXRhLCB3ZSBzZWUgdGhhdCBmb3IgMjAxMyBhbmQgMjAxNyB0aGVyZSBpcyBvbmx5IG9uZSBtb250aCBvZiBzYWxlcyBmb3IgZWFjaCBvZiB0aGVzZSB5ZWFycyAoMTIgZm9yIDIwMTMgYW5kIDEgZm9yIDIwMTcpLiBUaGVyZWZvcmUsIG5laXRoZXIgb2YgdGhlc2UgYXJlIGV2ZW5seSByZXByZXNlbnRlZC4gU2FsZXMgaGF2ZSBiZWVuIGluY3JlYXNpbmcgeWVhciBvdmVyIHllYXIsIHdpdGggMjAxNiBiZWluZyB0aGUgbGFyZ2VzdCBzYWxlcyB0byBkYXRlLiBBdCB0aGlzIHJhdGUsIHdlIG1pZ2h0IGV4cGVjdCAyMDE3IHRvIGhhdmUgdGhlIGxhcmdlc3Qgc2FsZXMuCgoyLiBXaGljaCAqbW9udGgqIGRpZCBQYXJjaCAmIFBvc2V5IGhhdmUgdGhlIGdyZWF0ZXN0IHNhbGVzIGluIHRlcm1zIG9mIHRvdGFsIGRvbGxhcnM/IEFyZSBhbGwgbW9udGhzIGV2ZW5seSByZXByZXNlbnRlZCBieSB0aGUgZGF0YXNldD8KCkluIG9yZGVyIGZvciB0aGlzIHRvIGJlICdmYWlyJywgd2Ugc2hvdWxkIHJlbW92ZSB0aGUgc2FsZXMgZnJvbSAyMDEzIGFuZCAyMDE3LiBGb3IgdGhlIHNhbWUgcmVhc29ucyBhcyBkaXNjdXNzZWQgYWJvdmUuCmBgYHtzcWx9ClNFTEVDVCBEQVRFX1BBUlQoJ21vbnRoJywgb2NjdXJyZWRfYXQpIG9yZF9tb250aCwgU1VNKHRvdGFsX2FtdF91c2QpIHRvdGFsX3NwZW50CkZST00gb3JkZXJzCldIRVJFIG9jY3VycmVkX2F0IEJFVFdFRU4gJzIwMTQtMDEtMDEnIEFORCAnMjAxNy0wMS0wMScKR1JPVVAgQlkgMQpPUkRFUiBCWSAyIERFU0M7CmBgYApUaGUgZ3JlYXRlc3Qgc2FsZXMgYW1vdW50cyBvY2N1ciBpbiBEZWNlbWJlciAoMTIpLgoKMy4gV2hpY2ggKnllYXIqIGRpZCBQYXJjaCAmIFBvc2V5IGhhdmUgdGhlIGdyZWF0ZXN0IHNhbGVzIGluIHRlcm1zIG9mIHRvdGFsIG51bWJlciBvZiBvcmRlcnM/IEFyZSBhbGwgeWVhcnMgZXZlbmx5IHJlcHJlc2VudGVkIGJ5IHRoZSBkYXRhc2V0PwpgYGB7c3FsfQpTRUxFQ1QgREFURV9QQVJUKCd5ZWFyJywgb2NjdXJyZWRfYXQpIG9yZF95ZWFyLCAgQ09VTlQoKikgdG90YWxfc2FsZXMKRlJPTSBvcmRlcnMKR1JPVVAgQlkgMQpPUkRFUiBCWSAyIERFU0M7CmBgYApBZ2FpbiwgMjAxNiBieSBmYXIgaGFzIHRoZSBtb3N0IGFtb3VudCBvZiBvcmRlcnMsIGJ1dCBhZ2FpbiAyMDEzIGFuZCAyMDE3IGFyZSBub3QgZXZlbmx5IHJlcHJlc2VudGVkIHRvIHRoZSBvdGhlciB5ZWFycyBpbiB0aGUgZGF0YXNldC4KCjQuIFdoaWNoICptb250aCogZGlkIFBhcmNoICYgUG9zZXkgaGF2ZSB0aGUgZ3JlYXRlc3Qgc2FsZXMgaW4gdGVybXMgb2YgdG90YWwgbnVtYmVyIG9mIG9yZGVycz8gQXJlIGFsbCBtb250aHMgZXZlbmx5IHJlcHJlc2VudGVkIGJ5IHRoZSBkYXRhc2V0PwpgYGB7c3FsfQpTRUxFQ1QgREFURV9QQVJUKCdtb250aCcsIG9jY3VycmVkX2F0KSBvcmRfbW9udGgsIENPVU5UKCopIHRvdGFsX3NhbGVzCkZST00gb3JkZXJzCldIRVJFIG9jY3VycmVkX2F0IEJFVFdFRU4gJzIwMTQtMDEtMDEnIEFORCAnMjAxNy0wMS0wMScKR1JPVVAgQlkgMQpPUkRFUiBCWSAyIERFU0M7CmBgYApEZWNlbWJlciBzdGlsbCBoYXMgdGhlIG1vc3Qgc2FsZXMsIGJ1dCBpbnRlcmVzdGluZ2x5LCBOb3ZlbWJlciBoYXMgdGhlIHNlY29uZCBtb3N0IHNhbGVzIChidXQgbm90IHRoZSBtb3N0IGRvbGxhciBzYWxlcy4gVG8gbWFrZSBhIGZhaXIgY29tcGFyaXNvbiBmcm9tIG9uZSBtb250aCB0byBhbm90aGVyIDIwMTcgYW5kIDIwMTMgZGF0YSB3ZXJlIHJlbW92ZWQuCgo1LiBJbiB3aGljaCBtb250aCBvZiB3aGljaCB5ZWFyIGRpZCBXYWxtYXJ0IHNwZW5kIHRoZSBtb3N0IG9uIGdsb3NzIHBhcGVyIGluIHRlcm1zIG9mIGRvbGxhcnM/CmBgYHtzcWx9ClNFTEVDVCBEQVRFX1RSVU5DKCdtb250aCcsIG8ub2NjdXJyZWRfYXQpIG9yZF9kYXRlLCBTVU0oby5nbG9zc19hbXRfdXNkKSB0b3Rfc3BlbnQKRlJPTSBvcmRlcnMgbyAKSk9JTiBhY2NvdW50cyBhCk9OIGEuaWQgPSBvLmFjY291bnRfaWQKV0hFUkUgYS5uYW1lID0gJ1dhbG1hcnQnCkdST1VQIEJZIDEKT1JERVIgQlkgMiBERVNDCkxJTUlUIDE7CmBgYApNYXkgMjAxNiB3YXMgd2hlbiBXYWxtYXJ0IHNwZW50IHRoZSBtb3N0IG9uIGdsb3NzIHBhcGVyLgoKIyMgQ0FTRSAKVGhlIENBU0Ugc3RhdGVtZW50IGFsd2F5cyBnb2VzIGluIHRoZSBTRUxFQ1QgY2xhdXNlLgpDQVNFIG11c3QgaW5jbHVkZSB0aGUgZm9sbG93aW5nIGNvbXBvbmVudHM6IFdIRU4sIFRIRU4sIGFuZCBFTkQuIEVMU0UgaXMgYW4gb3B0aW9uYWwgY29tcG9uZW50IHRvIGNhdGNoIGNhc2VzIHRoYXQgZGlkbuKAmXQgbWVldCBhbnkgb2YgdGhlIG90aGVyIHByZXZpb3VzIENBU0UgY29uZGl0aW9ucy4KWW91IGNhbiBtYWtlIGFueSBjb25kaXRpb25hbCBzdGF0ZW1lbnQgdXNpbmcgYW55IGNvbmRpdGlvbmFsIG9wZXJhdG9yIChsaWtlIFdIRVJFKSBiZXR3ZWVuIFdIRU4gYW5kIFRIRU4uIFRoaXMgaW5jbHVkZXMgc3RyaW5naW5nIHRvZ2V0aGVyIG11bHRpcGxlIGNvbmRpdGlvbmFsIHN0YXRlbWVudHMgdXNpbmcgQU5EIGFuZCBPUi4KWW91IGNhbiBpbmNsdWRlIG11bHRpcGxlIFdIRU4gc3RhdGVtZW50cywgYXMgd2VsbCBhcyBhbiBFTFNFIHN0YXRlbWVudCBhZ2FpbiwgdG8gZGVhbCB3aXRoIGFueSB1bmFkZHJlc3NlZCBjb25kaXRpb25zLgojIyMgRXhhbXBsZQpJbiBhIHF1aXogcXVlc3Rpb24gaW4gdGhlIHByZXZpb3VzIEJhc2ljIFNRTCBsZXNzb24sIHlvdSBzYXcgdGhpcyBxdWVzdGlvbjoKMS4gQ3JlYXRlIGEgY29sdW1uIHRoYXQgZGl2aWRlcyB0aGUgc3RhbmRhcmRfYW10X3VzZCBieSB0aGUgc3RhbmRhcmRfcXR5IHRvIGZpbmQgdGhlIHVuaXQgcHJpY2UgZm9yIHN0YW5kYXJkIHBhcGVyIGZvciBlYWNoIG9yZGVyLiBMaW1pdCB0aGUgcmVzdWx0cyB0byB0aGUgZmlyc3QgMTAgb3JkZXJzLCBhbmQgaW5jbHVkZSB0aGUgaWQgYW5kIGFjY291bnRfaWQgZmllbGRzLiAqTk9URSAtIHlvdSB3aWxsIGJlIHRocm93biBhbiBlcnJvciB3aXRoIHRoZSBjb3JyZWN0IHNvbHV0aW9uIHRvIHRoaXMgcXVlc3Rpb24uIFRoaXMgaXMgZm9yIGEgZGl2aXNpb24gYnkgemVyby4gWW91IHdpbGwgbGVhcm4gaG93IHRvIGdldCBhIHNvbHV0aW9uIHdpdGhvdXQgYW4gZXJyb3IgdG8gdGhpcyBxdWVyeSB3aGVuIHlvdSBsZWFybiBhYm91dCBDQVNFIHN0YXRlbWVudHMgaW4gYSBsYXRlciBzZWN0aW9uLioKTGV0J3Mgc2VlIGhvdyB3ZSBjYW4gdXNlIHRoZSBDQVNFIHN0YXRlbWVudCB0byBnZXQgYXJvdW5kIHRoaXMgZXJyb3IuCmBgYHtzcWx9ClNFTEVDVCBpZCwgYWNjb3VudF9pZCwgc3RhbmRhcmRfYW10X3VzZC9zdGFuZGFyZF9xdHkgQVMgdW5pdF9wcmljZQpGUk9NIG9yZGVycwpMSU1JVCAxMDsKYGBgCk5vdywgbGV0J3MgdXNlIGEgQ0FTRSBzdGF0ZW1lbnQuIFRoaXMgd2F5IGFueSB0aW1lIHRoZSBzdGFuZGFyZF9xdHkgaXMgemVybywgd2Ugd2lsbCByZXR1cm4gMCwgYW5kIG90aGVyd2lzZSB3ZSB3aWxsIHJldHVybiB0aGUgdW5pdF9wcmljZS4KYGBge3NxbH0KU0VMRUNUIGFjY291bnRfaWQsIENBU0UgV0hFTiBzdGFuZGFyZF9xdHkgPSAwIE9SIHN0YW5kYXJkX3F0eSBJUyBOVUxMIFRIRU4gMAogICAgICAgICAgICAgICAgICAgICAgICBFTFNFIHN0YW5kYXJkX2FtdF91c2Qvc3RhbmRhcmRfcXR5IEVORCBBUyB1bml0X3ByaWNlCkZST00gb3JkZXJzCkxJTUlUIDEwOwpgYGAKTm93IHRoZSBmaXJzdCBwYXJ0IG9mIHRoZSBzdGF0ZW1lbnQgd2lsbCBjYXRjaCBhbnkgb2YgdGhvc2UgZGl2aXNpb24gYnkgemVybyB2YWx1ZXMgdGhhdCB3ZXJlIGNhdXNpbmcgdGhlIGVycm9yLCBhbmQgdGhlIG90aGVyIGNvbXBvbmVudHMgd2lsbCBjb21wdXRlIHRoZSBkaXZpc2lvbiBhcyBuZWNlc3NhcnkuIFlvdSB3aWxsIG5vdGljZSwgd2UgZXNzZW50aWFsbHkgY2hhcmdlIGFsbCBvZiBvdXIgYWNjb3VudHMgNC45OSBmb3Igc3RhbmRhcmQgcGFwZXIuIEl0IG1ha2VzIHNlbnNlIHRoaXMgZG9lc24ndCBmbHVjdHVhdGUsIGFuZCBpdCBpcyBtb3JlIGFjY3VyYXRlIHRoYW4gYWRkaW5nIDEgaW4gdGhlIGRlbm9taW5hdG9yIGxpa2Ugb3VyIHF1aWNrIGZpeCBtaWdodCBoYXZlIGJlZW4gaW4gdGhlIGVhcmxpZXIgbGVzc29uLgoKIyMjIFF1ZXN0aW9ucwoxLiBXZSB3b3VsZCBsaWtlIHRvIHVuZGVyc3RhbmQgMyBkaWZmZXJlbnQgbGV2ZWxzIG9mIGN1c3RvbWVycyBiYXNlZCBvbiB0aGUgYW1vdW50IGFzc29jaWF0ZWQgd2l0aCB0aGVpciBwdXJjaGFzZXMuIFRoZSB0b3AgYnJhbmNoIGluY2x1ZGVzIGFueW9uZSB3aXRoIGEgTGlmZXRpbWUgVmFsdWUgKHRvdGFsIHNhbGVzIG9mIGFsbCBvcmRlcnMpIGdyZWF0ZXIgdGhhbiAyMDAsMDAwIHVzZC4gVGhlIHNlY29uZCBicmFuY2ggaXMgYmV0d2VlbiAyMDAsMDAwIGFuZCAxMDAsMDAwIHVzZC4gVGhlIGxvd2VzdCBicmFuY2ggaXMgYW55b25lIHVuZGVyIDEwMCwwMDAgdXNkLiBQcm92aWRlIGEgdGFibGUgdGhhdCBpbmNsdWRlcyB0aGUgbGV2ZWwgYXNzb2NpYXRlZCB3aXRoIGVhY2ggYWNjb3VudC4gWW91IHNob3VsZCBwcm92aWRlIHRoZSBhY2NvdW50IG5hbWUsIHRoZSB0b3RhbCBzYWxlcyBvZiBhbGwgb3JkZXJzIGZvciB0aGUgY3VzdG9tZXIsIGFuZCB0aGUgbGV2ZWwuIE9yZGVyIHdpdGggdGhlIHRvcCBzcGVuZGluZyBjdXN0b21lcnMgbGlzdGVkIGZpcnN0LgpgYGB7c3FsfQpTRUxFQ1QgYS5uYW1lLCBTVU0odG90YWxfYW10X3VzZCkgdG90YWxfc3BlbnQsIAogICAgIENBU0UgV0hFTiBTVU0odG90YWxfYW10X3VzZCkgPiAyMDAwMDAgVEhFTiAndG9wJwogICAgIFdIRU4gIFNVTSh0b3RhbF9hbXRfdXNkKSA+IDEwMDAwMCBUSEVOICdtaWRkbGUnCiAgICAgRUxTRSAnbG93JyBFTkQgQVMgY3VzdG9tZXJfbGV2ZWwKRlJPTSBvcmRlcnMgbwpKT0lOIGFjY291bnRzIGEKT04gby5hY2NvdW50X2lkID0gYS5pZCAKR1JPVVAgQlkgYS5uYW1lCk9SREVSIEJZIDIgREVTQzsKYGBgCgoyLiBXZSB3b3VsZCBub3cgbGlrZSB0byBwZXJmb3JtIGEgc2ltaWxhciBjYWxjdWxhdGlvbiB0byB0aGUgZmlyc3QsIGJ1dCB3ZSB3YW50IHRvIG9idGFpbiB0aGUgdG90YWwgYW1vdW50IHNwZW50IGJ5IGN1c3RvbWVycyBvbmx5IGluIDIwMTYgYW5kIDIwMTcuIEtlZXAgdGhlIHNhbWUgbGV2ZWxzIGFzIGluIHRoZSBwcmV2aW91cyBxdWVzdGlvbi4gT3JkZXIgd2l0aCB0aGUgdG9wIHNwZW5kaW5nIGN1c3RvbWVycyBsaXN0ZWQgZmlyc3QuCmBgYHtzcWx9ClNFTEVDVCBhLm5hbWUsIFNVTSh0b3RhbF9hbXRfdXNkKSB0b3RhbF9zcGVudCwgCiAgICAgQ0FTRSBXSEVOIFNVTSh0b3RhbF9hbXRfdXNkKSA+IDIwMDAwMCBUSEVOICd0b3AnCiAgICAgV0hFTiAgU1VNKHRvdGFsX2FtdF91c2QpID4gMTAwMDAwIFRIRU4gJ21pZGRsZScKICAgICBFTFNFICdsb3cnIEVORCBBUyBjdXN0b21lcl9sZXZlbApGUk9NIG9yZGVycyBvCkpPSU4gYWNjb3VudHMgYQpPTiBvLmFjY291bnRfaWQgPSBhLmlkCldIRVJFIG9jY3VycmVkX2F0ID4gJzIwMTUtMTItMzEnIApHUk9VUCBCWSAxCk9SREVSIEJZIDIgREVTQzsKYGBgCgozLiBXZSB3b3VsZCBsaWtlIHRvIGlkZW50aWZ5IHRvcCBwZXJmb3JtaW5nIHNhbGVzIHJlcHMsIHdoaWNoIGFyZSBzYWxlcyByZXBzIGFzc29jaWF0ZWQgd2l0aCBtb3JlIHRoYW4gMjAwIG9yZGVycy4gQ3JlYXRlIGEgdGFibGUgd2l0aCB0aGUgc2FsZXMgcmVwIG5hbWUsIHRoZSB0b3RhbCBudW1iZXIgb2Ygb3JkZXJzLCBhbmQgYSBjb2x1bW4gd2l0aCB0b3Agb3Igbm90IGRlcGVuZGluZyBvbiBpZiB0aGV5IGhhdmUgbW9yZSB0aGFuIDIwMCBvcmRlcnMuIFBsYWNlIHRoZSB0b3Agc2FsZXMgcGVvcGxlIGZpcnN0IGluIHlvdXIgZmluYWwgdGFibGUuCmBgYHtzcWx9ClNFTEVDVCBzLm5hbWUsIENPVU5UKCopIG51bV9vcmRzLAogICAgIENBU0UgV0hFTiBDT1VOVCgqKSA+IDIwMCBUSEVOICd0b3AnCiAgICAgRUxTRSAnbm90JyBFTkQgQVMgc2FsZXNfcmVwX2xldmVsCkZST00gb3JkZXJzIG8KSk9JTiBhY2NvdW50cyBhCk9OIG8uYWNjb3VudF9pZCA9IGEuaWQgCkpPSU4gc2FsZXNfcmVwcyBzCk9OIHMuaWQgPSBhLnNhbGVzX3JlcF9pZApHUk9VUCBCWSBzLm5hbWUKT1JERVIgQlkgMiBERVNDOwpgYGAKSXQgaXMgd29ydGggbWVudGlvbmluZyB0aGF0IHRoaXMgYXNzdW1lcyBlYWNoIG5hbWUgaXMgdW5pcXVlIC0gd2hpY2ggaGFzIGJlZW4gZG9uZSBhIGZldyB0aW1lcy4gV2Ugb3RoZXJ3aXNlIHdvdWxkIHdhbnQgdG8gYnJlYWsgYnkgdGhlIG5hbWUgYW5kIHRoZSBpZCBvZiB0aGUgdGFibGUuCgo0LiBUaGUgcHJldmlvdXMgZGlkbid0IGFjY291bnQgZm9yIHRoZSBtaWRkbGUsIG5vciB0aGUgZG9sbGFyIGFtb3VudCBhc3NvY2lhdGVkIHdpdGggdGhlIHNhbGVzLiBNYW5hZ2VtZW50IGRlY2lkZXMgdGhleSB3YW50IHRvIHNlZSB0aGVzZSBjaGFyYWN0ZXJpc3RpY3MgcmVwcmVzZW50ZWQgYXMgd2VsbC4gV2Ugd291bGQgbGlrZSB0byBpZGVudGlmeSB0b3AgcGVyZm9ybWluZyBzYWxlcyByZXBzLCB3aGljaCBhcmUgc2FsZXMgcmVwcyBhc3NvY2lhdGVkIHdpdGggbW9yZSB0aGFuIDIwMCBvcmRlcnMgb3IgbW9yZSB0aGFuIDc1MDAwMCBpbiB0b3RhbCBzYWxlcy4gVGhlIG1pZGRsZSBncm91cCBoYXMgYW55IHJlcCB3aXRoIG1vcmUgdGhhbiAxNTAgb3JkZXJzIG9yIDUwMDAwMCBpbiBzYWxlcy4gQ3JlYXRlIGEgdGFibGUgd2l0aCB0aGUgc2FsZXMgcmVwIG5hbWUsIHRoZSB0b3RhbCBudW1iZXIgb2Ygb3JkZXJzLCB0b3RhbCBzYWxlcyBhY3Jvc3MgYWxsIG9yZGVycywgYW5kIGEgY29sdW1uIHdpdGggdG9wLCBtaWRkbGUsIG9yIGxvdyBkZXBlbmRpbmcgb24gdGhpcyBjcml0ZXJpYS4gUGxhY2UgdGhlIHRvcCBzYWxlcyBwZW9wbGUgYmFzZWQgb24gZG9sbGFyIGFtb3VudCBvZiBzYWxlcyBmaXJzdCBpbiB5b3VyIGZpbmFsIHRhYmxlLiBZb3UgbWlnaHQgc2VlIGEgZmV3IHVwc2V0IHNhbGVzIHBlb3BsZSBieSB0aGlzIGNyaXRlcmlhIQpgYGB7c3FsfQpTRUxFQ1Qgcy5uYW1lLCBDT1VOVCgqKSwgU1VNKG8udG90YWxfYW10X3VzZCkgdG90YWxfc3BlbnQsIAogICAgIENBU0UgV0hFTiBDT1VOVCgqKSA+IDIwMCBPUiBTVU0oby50b3RhbF9hbXRfdXNkKSA+IDc1MDAwMCBUSEVOICd0b3AnCiAgICAgV0hFTiBDT1VOVCgqKSA+IDE1MCBPUiBTVU0oby50b3RhbF9hbXRfdXNkKSA+IDUwMDAwMCBUSEVOICdtaWRkbGUnCiAgICAgRUxTRSAnbG93JyBFTkQgQVMgc2FsZXNfcmVwX2xldmVsCkZST00gb3JkZXJzIG8KSk9JTiBhY2NvdW50cyBhCk9OIG8uYWNjb3VudF9pZCA9IGEuaWQgCkpPSU4gc2FsZXNfcmVwcyBzCk9OIHMuaWQgPSBhLnNhbGVzX3JlcF9pZApHUk9VUCBCWSBzLm5hbWUKT1JERVIgQlkgMyBERVNDOwpgYGAK